`

用HttpSessionListener与HttpSessionBindingListener实现在线人数统计

 
阅读更多

http://www.cnblogs.com/shencheng/archive/2011/01/07/1930227.html

 

在线人数统计方面的实现,上网找了下这方面的知识,最初我的想法是,管理session,如果session销毁了就减少,如果登陆用户了就新增一个,但是如果是用户非法退出,如:未注销,关闭浏览器等,这个用户的session是管理不到的,最后决定用HttpSessionListener接口或HttpSessionBindingListener接口来实现,通过监听session的新建和销毁来控制,详细如下。

先添加登陆的页面index.jsp

1
2
3
4
5
6
7
8
9
10
11
12
13
<%@ page contentType="text/html;charset=utf-8"%>
<html>
<head>
<title>test</title>             
</head>
<body>
<form action="login.jsp" method="post">
    用户名:<input type="text" name="username" />
    <br />
    <input type="submit" value="登录" />
</form>
</body>
</html>

点击登陆后跳转的login.jsp(为了方便,用jsp做servlet,同学们用的时候记得改过来)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<%@ page contentType="text/html;charset=utf-8"%>
<%@ page import="java.util.*"%>
<%
    request.setCharacterEncoding("UTF-8");
    // 取得登录的用户名
    String username = request.getParameter("username");
    // 把用户名保存进session
    session.setAttribute("username", username);
    // 把用户名放入在线列表
    List onlineUserList = (List) application.getAttribute("onlineUserList");
    // 第一次使用前,需要初始化
    if (onlineUserList == null) {
        onlineUserList = new ArrayList();
        application.setAttribute("onlineUserList", onlineUserList);
    }
    onlineUserList.add(username);
    // 成功
    response.sendRedirect("result.jsp");
%>

登陆成功跳转到显示页面result.jsp

1
2
3
<%@ page contentType="text/html;charset=utf-8"%>
<%@ page isELIgnored="false"%>
<%@page import="java.util.List"%>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<h3>您好:${username} [<a href="logout.jsp">注销</a>]</h3>
当前在线用户:
<table>
<%
    List onlineUserList = (List) application.getAttribute("onlineUserList");
    for (int i = 0; i < onlineUserList.size(); i++) {
    String onlineUsername = (String) onlineUserList.get(i);
%>
    <tr>
        <td><%=onlineUsername%></td>
    </tr>
<%
}
%>
</table>

点击注销页面logout.jsp页面

1
2
3
4
5
6
7
8
9
10
11
12
13
<%@ page contentType="text/html;charset=utf-8"%>
<%@ page import="java.util.*"%>
<%
    // 取得登录的用户名
    String username = (String) session.getAttribute("username");
    // 销毁session
    session.invalidate();
    // 从在线列表中删除用户名
    List onlineUserList = (List) application.getAttribute("onlineUserList");
    onlineUserList.remove(username);
    // 成功
    response.sendRedirect("index.jsp");
%>

OK,登陆、查看、注销页面都有了,下面开始新建监听器

1、HttpSessionListener

添加类OnlineUserListener,继承HttpSessionListener,HttpSessionListener中有两个方法sessionCreated(HttpSessionEvent event)与sessionDestroyed(HttpSessionEvent event),前者是监听session的新建,后者是监听session的销毁。

 OnlineUserListener代码如下:

 

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
package com.test;
 
import java.util.List;
import javax.servlet.ServletContext;
import javax.servlet.http.HttpSession;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;
/**
 * @author 版本
 */
public class OnlineUserListener implements HttpSessionListener {
 
    public void sessionCreated(HttpSessionEvent event) {
        System.out.println("新建session:"+event.getSession().getId());
    }
    public void sessionDestroyed(HttpSessionEvent event) {
        HttpSession session = event.getSession();
        ServletContext application = session.getServletContext();
        // 取得登录的用户名
        String username = (String) session.getAttribute("username");
        // 从在线列表中删除用户名
        List onlineUserList = (List) application.getAttribute("onlineUserList");
        onlineUserList.remove(username);
        System.out.println(username+"已经退出!");
    }
}

web.xml配置:

1
2
3
<listener>
  <listener-class>com.test.OnlineUserListener</listener-class>
</listener>

一旦监听器发现调用了sessionDestoryed方法就会把其用户从在线人数中delete,在下面两种情况下会发生sessionDestoryed事件

a.执行session.invalidate()方法时

logout.jsp中调用了 session.invalidate()方法

b.session会话超时

session的默认超时事件是30分钟,30分钟后自动销毁session

 

2、HttpSessionBindingListener

HttpSessionBindingListener虽然叫做监听器,但使用方法与HttpSessionListener完全不同。我们实际看一下它是如何使用的。

新建类OnlineUserBindingListener,实现HttpSessionBindingListener接口,构造方法传入username参数,HttpSessionBindingListener内有两个方法valueBound(HttpSessionBindingEvent event)和valueUnbound(HttpSessionBindingEvent event),前者为数据绑定,后者为取消绑定

所谓对session进行数据绑定,就是调用session.setAttribute()把HttpSessionBindingListener保存进session中。

在login.jsp中做这一步:

1
2
3
4
5
6
7
8
9
10
11
12
<%@page import="com.test.OnlineUserBindingListener"%>
<%@ page contentType="text/html;charset=utf-8"%>
<%@ page import="java.util.*"%>
<%
    request.setCharacterEncoding("UTF-8");
    // 取得登录的用户名
    String username = request.getParameter("username");
       // 把用户名放入在线列表
    session.setAttribute("onlineUserBindingListener", new OnlineUserBindingListener(username));
    // 成功
    response.sendRedirect("result.jsp");
%>

这就是HttpSessionBindingListener和HttpSessionListener之间的最大区别:HttpSessionListener只需要设置到web.xml中就可以监听整个应用中的所有session。HttpSessionBindingListener必须实例化后放入某一个session中,才可以进行监听。

从监听范围上比较,HttpSessionListener设置一次就可以监听所有session,HttpSessionBindingListener通常都是一对一的。

正是这种区别成就了HttpSessionBindingListener的优势,我们可以让每个listener对应一个username,这样就不需要每次再去session中读取username,进一步可以将所有操作在线列表的代码都移入listener,更容易维护。

HttpSessionBindingListener代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
package com.test;
 
import java.util.ArrayList;
import java.util.List;
import javax.servlet.ServletContext;
import javax.servlet.http.HttpSession;
import javax.servlet.http.HttpSessionBindingEvent;
import javax.servlet.http.HttpSessionBindingListener;
 
public class OnlineUserBindingListener implements HttpSessionBindingListener {
    String username;
     
    public OnlineUserBindingListener(String username){
        this.username=username;
    }
    public void valueBound(HttpSessionBindingEvent event) {
        HttpSession session = event.getSession();
        ServletContext application = session.getServletContext();
        // 把用户名放入在线列表
        List onlineUserList = (List) application.getAttribute("onlineUserList");
        // 第一次使用前,需要初始化
        if (onlineUserList == null) {
            onlineUserList = new ArrayList();
            application.setAttribute("onlineUserList", onlineUserList);
        }
        onlineUserList.add(this.username);
 
        //ServletContext application = arg0.getSession().getServletContext();

 //Set<Integer> onlineAdminuseridIdSet = (Set)application.getAttribute("onlineAdminuseridIdSet");

 //if(onlineAdminuseridIdSet == null){

//onlineAdminuseridIdSet = Sets.newTreeSet();

//application.setAttribute("onlineAdminuseridIdSet", onlineAdminuseridIdSet);

 //}

 //onlineAdminuseridIdSet.add(adminuserid);

    }
 
    public void valueUnbound(HttpSessionBindingEvent event) {
        HttpSession session = event.getSession();
        ServletContext application = session.getServletContext();
 
        // 从在线列表中删除用户名
        List onlineUserList = (List) application.getAttribute("onlineUserList");
        onlineUserList.remove(this.username);
        System.out.println(this.username + "退出。");
 
    }
 
}

这里可以直接使用listener的username操作在线列表,不必再去担心session中是否存在username。

valueUnbound的触发条件是以下三种情况:

a.执行session.invalidate()时。

b.session超时,自动销毁时。

c.执行session.setAttribute("onlineUserListener", "其他对象");或session.removeAttribute("onlineUserListener");将listener从session中删除时。

因此,只要不将listener从session中删除,就可以监听到session的销毁。

分享到:
评论

相关推荐

    jsp 统计在线人数利用HttpSessionListener、HttpSessionBindingListener

    jsp 统计在线人数实例,是个完整的项目实例,部署即可查看效果! 通过使用:HttpSessionListener、HttpSessionBindingListener来实现. valueBound 方法登录触发。 valueUnbound 方法退出触发。

    在线人数统计

    在这个小型案例中,我们使用了Java Servlet中的`HttpSessionListener`和`HttpSessionBindingListener`接口来实现这一功能。以下是对这两个接口以及整个案例的详细解释: 1. **`HttpSessionListener`接口**: `...

    servlet 监听器范例

    通过这个“servlet 监听器范例”,我们不仅学习了如何使用Servlet监听器来统计在线用户,还了解了如何跟踪用户登录状态。这个示例对于理解和实现在Java Web应用程序中进行用户管理、监控系统活动具有很大的价值。

    cmd脚本-bat批处理-删除右键“新建”菜单项目.zip

    cmd脚本-bat批处理-删除右键“新建”菜单项目.zip

    cmd脚本-bat批处理-YLMF系统服务优化.zip

    cmd脚本-bat批处理-YLMF系统服务优化.zip

    cmd-bat-批处理-脚本-替换系统主题.zip

    cmd-bat-批处理-脚本-替换系统主题.zip

    【数据库管理】MySQL索引技术详解:提升查询性能的关键方法与应用场景

    内容概要:本文详细介绍了 MySQL 索引的概念、类型及其创建、修改和删除方法。索引作为一种数据结构,能够显著提升数据库查询速度,类似于书籍的索引帮助快速定位信息。文章解释了单列索引和组合索引的区别,并强调了索引的创建应基于 SQL 查询的条件部分。此外,文中还探讨了普通索引和唯一索引的特点及创建方式,包括使用 `CREATE INDEX`、`ALTER TABLE` 和 `CREATE TABLE` 语句的具体用法。最后,文章提醒用户注意索引的维护成本,如额外的存储空间和对插入、更新和删除操作的影响,以及如何通过 `SHOW INDEX` 查看索引信息。 适用人群:适用于数据库管理员、软件开发人员以及对 MySQL 数据库优化感兴趣的用户。 使用场景及目标:①需要优化数据库查询性能,特别是处理大型数据表时;②理解不同类型的索引(如普通索引、唯一索引)及其应用场景;③掌握索引的创建、修改和删除方法,确保数据库的高效运行。 其他说明:创建索引虽能提高查询效率,但也需权衡其带来的额外开销。建议在实际应用中根据具体需求合理规划索引,避免过度使用导致性能下降。同时,定期检查和优化现有索引,确保其始终符合业务需求。

    Android校园二手交易App项目源码(高分期末大作业)

    Android校园二手交易App项目源码(高分期末大作业),个人经导师指导并认可通过的高分设计项目,评审分98分,项目中的源码都是经过本地编译过可运行的,都经过严格调试,确保可以运行!主要针对计算机相关专业的正在做大作业、毕业设计的学生和需要项目实战练习的学习者,资源项目的难度比较适中,内容都是经过助教老师审定过的能够满足学习、使用需求,如果有需要的话可以放心下载使用。 Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)Android校园二手交易App项目源码(高分期末大作业)And

    cmd脚本-bat批处理-RUN.zip

    cmd脚本-bat批处理-RUN.zip

    cmd-bat-批处理-脚本-倒记时(全屏).zip

    cmd-bat-批处理-脚本-倒记时(全屏).zip

    cmd-bat-批处理-脚本-老外写的系统文件替换.zip

    cmd-bat-批处理-脚本-老外写的系统文件替换.zip

    cmd脚本-bat批处理-去最大最小值后求剩余数组平均值.zip

    cmd脚本-bat批处理-去最大最小值后求剩余数组平均值.zip

    cmd-bat-批处理-脚本-弹出对话框.zip

    cmd-bat-批处理-脚本-弹出对话框.zip

    【数据库管理】MySQL数据表完整复制方法详解:包括结构与数据迁移操作步骤在MySQL中如何

    内容概要:本文详细介绍了如何在MySQL中完整复制数据表的方法。首先,通过SHOW CREATE TABLE命令获取原表的创建语句,包括表结构、索引等信息;接着,修改语句中的表名并执行,以创建新的空表;最后,利用INSERT INTO...SELECT语句将源表的数据插入到新表中,从而实现表结构与数据的全面复制。此外,还介绍了使用mysqldump命令进行表级备份和恢复的操作,即先用mysqldump导出表结构和数据到SQL文件,再通过mysql命令将此文件导入到目标数据库中,适用于跨库或跨服务器的数据迁移。; 适合人群:数据库管理员、开发人员以及需要对MySQL数据库进行表复制操作的技术人员。; 使用场景及目标:①当需要在同一数据库内创建具有相同结构和数据的新表时;②在不同数据库之间迁移单个表的数据和结构;③作为数据备份的一种手段,确保重要表数据的安全性。; 其他说明:文中提供的方法不仅限于简单的表复制,在实际应用中还可以根据需求调整表结构或筛选特定数据行进行复制。同时,使用mysqldump工具时要注意用户名、密码及数据库名称等参数的正确性。

    cmd-bat-批处理-脚本-OptimizeXp.zip

    cmd-bat-批处理-脚本-OptimizeXp.zip

    【数据库管理】MySQL处理重复数据的技术方法:确保数据唯一性及重复数据的统计与删除

    内容概要:本文档主要介绍了MySQL中处理重复数据的方法。首先,讲解了如何通过设置主键(PRIMARY KEY)或唯一索引(UNIQUE)来防止数据表中出现重复数据,包括创建具有唯一约束条件的表结构示例。接着,阐述了两种特殊的插入语句——`INSERT IGNORE INTO`和`REPLACE INTO`的区别及其应用场景,前者用于忽略重复数据插入,后者则会替换已有的重复记录。然后,详细描述了如何使用SQL语句统计表内的重复记录,以及如何利用`DISTINCT`关键字或`GROUP BY`子句来过滤并读取不重复的数据。最后,提供了几种删除数据表中重复数据的方法,如创建临时表再重建原表、直接修改表结构添加唯一性约束等。 适合人群:有一定SQL基础,对MySQL数据库管理和优化感兴趣的开发人员、数据库管理员。 使用场景及目标:①掌握通过设置主键或唯一索引来避免数据重复;②学会使用特定的SQL语句处理重复数据,包括统计、过滤和删除;③理解不同插入语句在面对重复数据时的行为差异。 阅读建议:在学习过程中,建议结合实际的MySQL环境进行练习,尤其是尝试文中提供的SQL语句,以便更好地理解和掌握处理重复数据的各种技巧。

    cmd-bat-批处理-脚本-关闭局域网共享.zip

    cmd-bat-批处理-脚本-关闭局域网共享.zip

    cmd-bat-批处理-脚本-比较两个文件夹下文件名的异同.zip

    cmd-bat-批处理-脚本-比较两个文件夹下文件名的异同.zip

    1743390592614.osm

    1743390592614.osm

    振动压路机振动轮.rar

    振动压路机振动轮.rar

Global site tag (gtag.js) - Google Analytics