知识点:HttpSessionBindingListener接口介绍
如果一个对象实现了HttpSessionBindingListener接口,当这个对象被绑定到Session中或者从session中被删除时,Servlet容器会通知这个对象,而这个对象在接收到通知后,可以做一些初始化或清除状态的操作。
javax.servlet.http.HttpSessionBindingListener接口提供了以下方法:
public
void valueBound(HttpSessionBindingEvent event)
当对象正在被绑定到Session中,Servlet容器调用这个方法来通知该对象。
public
void valueUnbound(HttpSessionBindingEvent event)
当从Session中删除对象时,Servlet容器调用这个方法来实现了HttpSessionBindingListener接口的对象,而这个对象
可以利用HttpSessionBindingEvent对象来访问与它相联系的HttpSession对象。 Javax.Servlet.http.HttpSessionBindingEvent类提供了以下两种方法。
public
HttpSessionBindingEvent(HttpSession session,java.lang.String
name)
public
HttpSessionBindingEvent(HttpSession session,java.lang.string
name,java.lang.Object value)
上面两个构造一个事件对象,当一个对象被绑定到Session中或者从Session中被删除时,用这个事件对象来通知它。
public
java.lang.String getName()
返回绑定到Session中或者从session中删除的属性的名字。
public
java.lang.Object getValue()
返回添加、删除或替换的属性的值。如果属性被添加或者被删除,这个方法返回属性的值。如果属性被替换,这个方法返回属性先前的值。
public
HttpSession getSession()
返回HttpSession对象。
下午比较闲(其实今天都很闲),想了一下在线人数统计方面的实现,上网找了下这方面的知识,最初我的想法是,管理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;
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的销毁。