监听器应用【统计网站人数、自定义session扫描器、踢人小案例】(修订版)

news/2024/5/19 20:58:58/文章来源:https://blog.csdn.net/Java_3y/article/details/98867949

前言

只有光头才能变强。

文本已收录至我的GitHub仓库,欢迎Star:https://github.com/ZhongFuCheng3y/3y

从第一篇已经讲解过了监听器的基本概念,以及Servlet各种的监听器。这篇博文主要讲解的是监听器的应用。

统计网站在线人数

ps:这个可以使用WebSocket来做,但这里讲解的是监听器,所以这里以监听器来举例子!

分析

我们在网站中一般使用Session来标识某用户是否登陆了,如果登陆了,就在Session域中保存相对应的属性。如果没有登陆,那么Session的属性就应该为空。

现在,我们想要统计的是网站的在线人数。我们应该这样做:我们监听是否有新的Session创建了,如果新创建了Sesssion,那么在线人数就应该+1。这个在线人数是整个站点的,所以应该有Context对象保存。

大致思路:

代码

public class CountOnline implements HttpSessionListener {    public void sessionCreated(HttpSessionEvent se) {        //获取得到Context对象,使用Context域对象保存用户在线的个数        ServletContext context = se.getSession().getServletContext();        //直接判断Context对象是否存在这个域,如果存在就人数+1,如果不存在,那么就将属性设置到Context域中        Integer num = (Integer) context.getAttribute("num");        if (num == null) {            context.setAttribute("num", 1);        } else {            num++;            context.setAttribute("num", num);        }    }    public void sessionDestroyed(HttpSessionEvent se) {        ServletContext context = se.getSession().getServletContext();        Integer num = (Integer) se.getSession().getAttribute("num");        if (num == null) {            context.setAttribute("num", 1);        } else {            num--;            context.setAttribute("num", num);        }    }}class CountOnline implements HttpSessionListener {

    public void sessionCreated(HttpSessionEvent se) {

        //获取得到Context对象,使用Context域对象保存用户在线的个数
        ServletContext context = se.getSession().getServletContext();

        //直接判断Context对象是否存在这个域,如果存在就人数+1,如果不存在,那么就将属性设置到Context域中
        Integer num = (Integer) context.getAttribute("num");

        if (num == null) {
            context.setAttribute("num"1);
        } else {
            num++;
            context.setAttribute("num", num);
        }
    }
    public void sessionDestroyed(HttpSessionEvent se) {

        ServletContext context = se.getSession().getServletContext();
        Integer num = (Integer) se.getSession().getAttribute("num");

        if (num == null) {
            context.setAttribute("num"1);
        } else {
            num--;
            context.setAttribute("num", num);
        }
    }
}
在线人数:${num}

测试

我们每使用一个浏览器访问服务器,都会新创建一个Session。那么网站的在线人数就会+1。

使用同一个页面刷新,还是使用的是那个Sesssion,所以网站的在线人数是不会变的。

640?wx_fmt=gif


自定义Session扫描器

我们都知道Session是保存在内存中的,如果Session过多,服务器的压力就会非常大。

但是呢,Session的默认失效时间是30分钟(30分钟没人用才会失效),这造成Seesion可能会过多(没人用也存在内存中,这不是明显浪费吗?)

当然啦,我们可以在web.xml文件中配置Session的生命周期。但是呢,这是由服务器来做的,我嫌它的时间不够准确。(有时候我配置了3分钟,它用4分钟才帮我移除掉Session)

所以,我决定自己用程序手工移除那些长时间没人用的Session。

分析

要想移除长时间没人用的Session,肯定要先拿到全部的Session啦。所以我们使用一个容器来装载站点所有的Session。。

只要Sesssion一创建了,就把Session添加到容器里边。毫无疑问的,我们需要监听Session了。

接着,我们要做的就是隔一段时间就去扫描一下全部Session,如果有Session长时间没使用了,我们就把它从内存中移除。隔一段时间去做某事,这肯定是定时器的任务呀。

定时器应该在服务器一启动的时候,就应该被创建了。因此还需要监听Context

最后,我们还要考虑到并发的问题,如果有人同时访问站点,那么监听Session创建的方法就会被并发访问了定时器扫描容器的时候,可能是获取不到所有的Session的

这需要我们做同步

于是乎,我们已经有大致的思路了


代码

public class Listener1 implements ServletContextListener,        HttpSessionListener {    //服务器一启动,就应该创建容器。我们使用的是LinkList(涉及到增删)。容器也应该是线程安全的。    List<HttpSession> list = Collections.synchronizedList(new LinkedList<HttpSession>());    //定义一把锁(Session添加到容器和扫描容器这两个操作应该同步起来)    private Object lock = 1;    public void contextInitialized(ServletContextEvent sce) {        Timer timer = new Timer();        //执行我想要的任务,0秒延时,每10秒执行一次        timer.schedule(new MyTask(list, lock), 0, 10 * 1000);    }    public void sessionCreated(HttpSessionEvent se) {        //只要Session一创建了,就应该添加到容器中        synchronized (lock) {            list.add(se.getSession());        }        System.out.println("Session被创建啦");    }    public void sessionDestroyed(HttpSessionEvent se) {        System.out.println("Session被销毁啦。");    }    public void contextDestroyed(ServletContextEvent sce) {    }}class Listener1 implements ServletContextListener,
        HttpSessionListener 
{



    //服务器一启动,就应该创建容器。我们使用的是LinkList(涉及到增删)。容器也应该是线程安全的。
    List<HttpSession> list = Collections.synchronizedList(new LinkedList<HttpSession>());

    //定义一把锁(Session添加到容器和扫描容器这两个操作应该同步起来)
    private Object lock = 1;

    public void contextInitialized(ServletContextEvent sce) {


        Timer timer = new Timer();
        //执行我想要的任务,0秒延时,每10秒执行一次
        timer.schedule(new MyTask(list, lock), 010 * 1000);

    }
    public void sessionCreated(HttpSessionEvent se) {

        //只要Session一创建了,就应该添加到容器中
        synchronized (lock) {
            list.add(se.getSession());
        }
        System.out.println("Session被创建啦");

    }

    public void sessionDestroyed(HttpSessionEvent se) {
        System.out.println("Session被销毁啦。");
    }
    public void contextDestroyed(ServletContextEvent sce) {

    }
}
/** 在任务中应该扫描容器,容器在监听器上,只能传递进来了。** 要想得到在监听器上的锁,也只能是传递进来** */class MyTask extends TimerTask {    private List<HttpSession> sessions;    private Object lock;    public MyTask(List<HttpSession> sessions, Object lock) {        this.sessions = sessions;        this.lock = lock;    }    @Override    public void run() {        synchronized (lock) {            //遍历容器            for (HttpSession session : sessions) {                //只要15秒没人使用,我就移除它啦                if (System.currentTimeMillis() - session.getLastAccessedTime() > (1000 * 15)) {                    session.invalidate();                    sessions.remove(session);                }            }        }    }}
class MyTask extends TimerTask {

    private List<HttpSession> sessions;
    private Object lock;

    public MyTask(List<HttpSession> sessions, Object lock) {
        this.sessions = sessions;
        this.lock = lock;
    }

    @Override
    public void run() {

        synchronized (lock) {
            //遍历容器
            for (HttpSession session : sessions) {

                //只要15秒没人使用,我就移除它啦
                if (System.currentTimeMillis() - session.getLastAccessedTime() > (1000 * 15)) {
                    session.invalidate();
                    sessions.remove(session);
                }

            }
        }
    }
}

15秒如果Session没有活跃,那么就被删除!

640?wx_fmt=gif


踢人小案列

列出所有的在线用户,后台管理者拥有踢人的权利,点击踢人的超链接,该用户就被注销了。

分析

首先,怎么能列出所有的在线用户呢??一般我们在线用户都是用Session来标记的,所有的在线用户就应该用一个容器来装载所有的Session。。

我们监听Session的是否有属性添加(监听Session的属性有添加、修改、删除三个方法。如果监听到Session添加了,那么这个肯定是个在线用户!)。

装载Session的容器应该是在Context里边的【属于全站点】,并且容器应该使用Map集合【待会还要通过用户的名字来把用户踢了】

思路:

代码

public class KickPerson implements HttpSessionAttributeListener {    // Public constructor is required by servlet spec    public KickPerson() {    }    public void attributeAdded(HttpSessionBindingEvent sbe) {        //得到context对象,看看context对象是否有容器装载Session        ServletContext context = sbe.getSession().getServletContext();        //如果没有,就创建一个呗        Map map = (Map) context.getAttribute("map");        if (map == null) {            map = new HashMap();            context.setAttribute("map", map);        }        //---------------------------------------------------------------------------------------        //得到Session属性的值        Object o = sbe.getValue();        //判断属性的内容是否是User对象        if (o instanceof User) {            User user = (User) o;            map.put(user.getUsername(), sbe.getSession());        }    }    public void attributeRemoved(HttpSessionBindingEvent sbe) {      /* This method is called when an attribute         is removed from a session.      */    }    public void attributeReplaced(HttpSessionBindingEvent sbe) {      /* This method is invoked when an attibute         is replaced in a session.      */    }}class KickPerson implements HttpSessionAttributeListener {

    // Public constructor is required by servlet spec
    public KickPerson() {
    }

    public void attributeAdded(HttpSessionBindingEvent sbe) {

        //得到context对象,看看context对象是否有容器装载Session
        ServletContext context = sbe.getSession().getServletContext();

        //如果没有,就创建一个呗
        Map map = (Map) context.getAttribute("map");
        if (map == null) {
            map = new HashMap();
            context.setAttribute("map", map);
        }

        //---------------------------------------------------------------------------------------

        //得到Session属性的值
        Object o = sbe.getValue();

        //判断属性的内容是否是User对象
        if (o instanceof User) {
            User user = (User) o;
            map.put(user.getUsername(), sbe.getSession());
        }
    }

    public void attributeRemoved(HttpSessionBindingEvent sbe) {
      /* This method is called when an attribute
         is removed from a session.
      */

    }

    public void attributeReplaced(HttpSessionBindingEvent sbe) {
      /* This method is invoked when an attibute
         is replaced in a session.
      */

    }
}
<form action="${pageContext.request.contextPath }/LoginServlet" method="post">    用户名:<input type="text" name="username">    <input type="submit" value="登陆"></form>
    用户名:<input type="text" name="username">
    <input type="submit" value="登陆">
</form>
//得到传递过来的数据String username = request.getParameter("username");User user = new User();user.setUsername(username);//标记该用户登陆了!request.getSession().setAttribute("user", user);//提供界面,告诉用户登陆是否成功request.setAttribute("message", "恭喜你,登陆成功了!");request.getRequestDispatcher("/message.jsp").forward(request, response);
String username = request.getParameter("username");

User user = new User();
user.setUsername(username);

//标记该用户登陆了!
request.getSession().setAttribute("user", user);

//提供界面,告诉用户登陆是否成功
request.setAttribute("message""恭喜你,登陆成功了!");
request.getRequestDispatcher("/message.jsp").forward(request, response);
<c:forEach items="${map}" var="me">    ${me.key} <a href="${pageContext.request.contextPath}/KickPersonServlet?username=${me.key}">踢了他吧</a>    <br></c:forEach>"me">

    ${me.key} <a href="${pageContext.request.contextPath}/KickPersonServlet?username=${me.key}">踢了他吧</a>

    <br>
</c:forEach>
String username = request.getParameter("username");//得到装载所有的Session的容器Map map = (Map) this.getServletContext().getAttribute("map");//通过名字得到SessionHttpSession httpSession = (HttpSession) map.get(username);httpSession.invalidate();map.remove(username);//摧毁完Session后,返回列出在线用户页面request.getRequestDispatcher("/listUser.jsp").forward(request, response);

//得到装载所有的Session的容器
Map map = (Map) this.getServletContext().getAttribute("map");

//通过名字得到Session
HttpSession httpSession = (HttpSession) map.get(username);
httpSession.invalidate();
map.remove(username);

//摧毁完Session后,返回列出在线用户页面
request.getRequestDispatcher("/listUser.jsp").forward(request, response);

测试

使用多个浏览器登陆来模拟在线用户(同一个浏览器使用的都是同一个Session)

640?wx_fmt=gif


监听Seesion的创建和监听Session属性的变化有啥区别???

最后

乐于输出干货的Java技术公众号:Java3y。公众号内有200多篇原创技术文章、海量视频资源、精美脑图,不妨来关注一下!

640?wx_fmt=jpeg

有帮助?好看!转发!640


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_725919.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

大型分布式网站架构设计与实践

阅读文本大概需要3分钟。SOA和RPC随着互联网规模发展&#xff0c;面向服务的体系架构(SOA)成为主流的架构方式&#xff0c;SOA的本质思想是高内聚、低耦合地实现分治&#xff0c;各个系统之间通过服务的方式进行交互&#xff0c;这样保证了交互的标准性&#xff0c;这对于一个复…

为什么有些网站域名不以www开头?什么是二级域名?

本文公众号来源&#xff1a;漫话编程 作者&#xff1a;漫话编程我在初学的时候不清楚啥叫顶级域名&#xff0c;啥是二级域名以及为啥我们输入网站地址的时候可以省略www&#xff01;这篇文章通俗易懂地告诉你&#xff01;38女神节那一天&#xff0c;女朋友问我淘宝网国际站的…

大型网站技术架构剖析

阅读文本大概需要3分钟。大型网站系统特点高并发&#xff0c;大流量高可用海量数据用户分布广泛&#xff0c;网络情况复杂安全环境恶劣需求快速变更&#xff0c;发布频繁渐进式发展大型网站发展演化过程一台服务器特点&#xff1a;没人应用程序、数据库、文件都在一个服务器。应…

[网站优化实战]公共CDN库/Nginx启用Gzip/全站CDN加速

本文公众号来源&#xff1a;Rude3Knife 作者&#xff1a;蛮三刀把刀网站加载优化的过程&#xff08;主要针对静态资源&#xff09;&#xff0c;思路可以借鉴一下&#xff01;前言接触到CDN的起因&#xff1a;我自己搭建的网站https://price.monitor4all.cn/网页打开的速度一直…

《大型网站系统与Java中间件》读书笔记(上)

前言 只有光头才能变强。 文本已收录至我的GitHub仓库&#xff0c;欢迎Star&#xff1a;https://github.com/ZhongFuCheng3y/3y 这本书买了一段时间了&#xff0c;之前在杭州没带过去&#xff0c;现在读完第三章&#xff0c;来做做笔记 这本书前三章都在科普和回顾中间件/分布式…

《大型网站系统与Java中间件》读书笔记 (二)

前言只有光头才能变强。文本已收录至我的GitHub仓库&#xff0c;欢迎Star&#xff1a;https://github.com/ZhongFuCheng3y/3y回顾上一篇&#xff1a;《大型网站系统与Java中间件》读书笔记&#xff08;一&#xff09;这周周末读了第四章&#xff0c;现在过来做做笔记&#xff0…

《大型网站系统与Java中间件》读书笔记 (中)

前言 只有光头才能变强。 文本已收录至我的GitHub仓库&#xff0c;欢迎Star&#xff1a;https://github.com/ZhongFuCheng3y/3y 回顾上一篇&#xff1a; 《大型网站系统与Java中间件》读书笔记&#xff08;一&#xff09; 这周周末读了第四章&#xff0c;现在过来做做笔记&am…

几个简单步骤可以提高网站浏览体验

摘要&#xff1a; STEP 1确定网站定位 你的网站定位是要以企业形象官网为主&#xff0c;或是以单纯贩售商品的购物网站&#xff0c;还是要跟使用者做互动活动的网站&#xff1f; 每个网站的目的都不尽相同&#xff0c;形象官网&#xff0c;可能需要着重于品牌理念、品牌故事、…

一次面试引发的思考(中小型网站优化思考)

前言 故事的起因是这样的&#xff0c;由于本人地处偏僻工作地点在美丽的冰城哈尔滨虽然地方很美丽&#xff0c;但是这里的软件行业实在是算不上“美丽”&#xff0c;这么多年由于个人原因或者公司原因经常换工作&#xff0c;因为这里都是中小型公司&#xff0c;没有什么大公司。…

我采访了同事,让他掏出了每天都会浏览的干货网站...这几个网站也太牛了吧!

前言 在周六的晚上&#xff0c;我日常去到公司写文章。想写一篇程序员常浏览的网站&#xff0c;刚好同事在我后面看我在干什么。于是我就对他进行了采访&#xff0c;问了一下他常去的网站有哪些。 这次我采访的是鸡蛋&#xff0c;他跟我一样大&#xff0c;但是技术比我优秀实在…

Django项目实践4 - Django网站管理(后台管理员)

http://blog.csdn.net/pipisorry/article/details/45079751 上篇&#xff1a;Django项目实践3 - Django模型 Introduction 对于某一类站点&#xff0c; 管理界面 是基础设施中很重要的一部分。这是以网页和有限的可信任管理者为基础的界面&#xff0c;它能够让你加入&#xff0…

亿级流量网站构架核心技术

高并发原则无状态拆分系统维度&#xff1a;根据系统功能/业务进行拆分功能维度&#xff1a;对一个系统进行功能再拆分读写维度&#xff1a;根据读写比例进行拆分AOP维度&#xff1a;根据访问特征模块维度&#xff1a;比如按照基础或代码维护特征进行拆分服务化&#xff1a;进程…

Python教程 - 廖雪峰的官方网站

2019独角兽企业重金招聘Python工程师标准>>> https://www.liaoxuefeng.com/ 转载于:https://my.oschina.net/u/3563297/blog/1622686

这也许是破解所有网站

您还担心各种资源获取不到吗&#xff1f;这里聚集了广大网友的智慧结晶所在&#xff01; 现在的技术真的是越来厉害了&#xff0c;而且相比于以往复杂的操作 现在的黑科技仿佛特别“亲民” 比如 我之前发过在“baidu”后面加“wp” 就能高速下载百度云资源 而且自己不需要…

thinkphp5项目--企业单车网站(五)

thinkphp5项目--企业单车网站&#xff08;五&#xff09; 项目地址 fry404006308/BicycleEnterpriseWebsite: Bicycle Enterprise Websitehttps://github.com/fry404006308/BicycleEnterpriseWebsite 一、BeyondAdmin使用 1、里面的功能真的超级全&#xff0c;连编辑器都有&…

使用Nginx实现多台服务器网站负载均衡的配置方法介绍

使用Nginx实现网站负载均衡的配置方法介绍1.准备工作1.1 Linux系统安装Nginx1.2 准备三台服务器1.3配置主服务器nginx访问路径1.4访问主服务器2.Nginx负载均衡的几种不同方式介绍2.1 轮询2.2 权重2.3 iphash2.4 最少连接2.5 fair服务器的响应时间来分配3.Nginx配置1.准备工作 …

Laravel访问网站页面空白

配置好了Laravel之后进行网站访问&#xff0c;发现是空白页面&#xff0c;第一次使用Laravel很迷茫。使用fiddler查看的时候出现500错误&#xff0c;网上查了下是因为根目录下的storage目录没有777权限&#xff0c;如图&#xff1a; 解决方法&#xff1a;使用chmod -R 777 s…

访问网站的时候出现Discuz! Database Error (2002) notconnect错误

自己用dz做的网站一直好好的&#xff0c;今天访问的时候出现了&#xff1a;Discuz! Database Error (2002) notconnect错误&#xff0c;如图&#xff1a; 解决方法&#xff1a; 修改/config/config_global.php中的config[‘db′][‘1′][‘dbhost′]‘localhost′;改为&…

知乎有哪些适合大学生浏览的网站?

2019独角兽企业重金招聘Python工程师标准>>> 作者&#xff1a;Adam 链接&#xff1a;https://www.zhihu.com/question/20136746/answer/299592153 来源&#xff1a;知乎 著作权归作者所有。商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处。 公开课 …

SEO新站与老站,如何解决网站降权问题?

针对网站降权的问题&#xff0c;百度与谷歌的评定标准基本相同&#xff0c;都是完全基于搜索引擎的用户体验&#xff0c;进行基础性的判断。简单理解&#xff1a;网站被降权&#xff0c;实际上是被搜索引擎降低信任评级的一个过程&#xff0c;那么你一定触碰了相关算法的识别机…