作者1:try出爱的异常
作者2:长安散人
下午比较闲(其实今天都很闲),想了一下在线人数统计方面的实现,上网找了下这方面的知识,最初我的想法是,管理session,如果session销毁了就减少,如果登陆用户了就新增一个,但是如果是用户非法退出,如:未注销,关闭浏览器等,这个用户的session是管理不到的,最后决定用HttpSessionListener接口或HttpSessionBindingListener接口来实现,通过监听session的新建和销毁来控制,详细如下。
先添加登陆的页面index.jsp
<%@ 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,同学们用的时候记得改过来)
<%@ 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
<%@ page contentType="text/html;charset=utf-8"%><%@ page isELIgnored="false"%><%@page import="java.util.List"%><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页面
<%@ 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代码如下:
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配置:
<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中做这一步:
<%@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代码如下:
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);}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的销毁。
附注: Java设置session超时(失效)的三种方式
1. 在web容器中设置(此处以tomcat为例)
在tomcat-5.0.28\conf\web.xml中设置,以下是tomcat 5.0中的默认配置:
- <!-- ==================== Default Session Configuration ================= -->
- <!-- You can set the default session timeout (in minutes) for all newly -->
- <!-- created sessions by modifying the value below. -->
- <session-config>
- <session-timeout>30</session-timeout>
- </session-config>
Tomcat默认session超时时间为30分钟,可以根据需要修改,负数或0为不限制session失效时间。
2. 在工程的web.xml中设置
- <!-- 时间单位为分钟 -->
<session-config>
<session-timeout>15</session-timeout>
</session-config>
3. 通过Java代码设置
session.setMaxInactiveInterval(30*60);//以秒为单位
三种方式优先级:1 < 2 <3
用HttpSessionListener与HttpSessionBindingListener实现在线人数统计
原文地址:http://www.cnblogs.com/lxl57610/p/7472670.html