在码农的世界里只有bug才能让人成长,The more bugs you encounter, the more efficient you will be!
java中的监听器分为三种:ServletRequestListener、HttpSessionListener、ServletContextListener;
ServletRequest--请求,在客户端对服务器发生请求(访问)时发生,请求一次发生一次(不考虑请求转发),
请求结束,该对象销毁;
HttpSession--会话,单个用户在某段时间内(Tomcat默认为30min,可以设置),自第一次发生请求(访问)
时创建,直到过期,该对象销毁;
ServletContext--上下文,该对象存在于服务器端,自服务器开启时创建,直到服务器关闭该对象销毁。
那么,就有这样一个思路:一个用户持续对网站的访问,请求数会很多,但是该用户多次请求的Session却一
直都是一个(唯一性)。我们可以在第一次访问时把Session对象的ID存到服务器(上下文),后面每一次请求的
时候,我们可以检查,该请求的Session对象是否已经在服务器(上下文),如果在,说明这次请求和之前的某次
请求是同一个用户,如果不在,那么将这个请求产生的新Session对象的ID存到服务器(上下文)。Session对象到
期,自然就被销毁(可以看做用户长时间每活动,*下线了),那么将这个SessionID所在的整条信息删除,也
就是可以看做,当前访问用户减少一个。其实除了多次请求产生的Session具有唯一性之外,发出请求的设备也是
具有唯一性的,我们可以通过获得设备的IP进行标识。我们可以将这两个信息存到用户类中,每个用户对象都有这
两个属性,并且是唯一的。
1.ServletRequest监听器
public class Requestlisen implements ServletRequestListener { public void requestDestroyed(ServletRequestEvent arg0) {
// TODO Auto-generated method stub } public void requestInitialized(ServletRequestEvent arg0) {
// TODO Auto-generated method stub
System.out.println("-----request创建了-------");
HttpServletRequest request = (HttpServletRequest) arg0
.getServletRequest();
if (arg0.getServletContext().getAttribute("userlist") == null) {
List<User> userlist = new ArrayList<User>();
arg0.getServletContext().setAttribute("userlist", userlist);
}
List<User> userlist = (List<User>) arg0.getServletContext()
.getAttribute("userlist");
String id = request.getSession().getId();
User user = (User) Getuserbyid.finduser(userlist, id);
if (user == null) {
System.out.println("-----当前请求对象不存在于服务器-----");
System.out.println("------将当前用户添加到服务器-----");
User newuser = new User();
newuser.setIp(request.getRemotePort() + "");
newuser.setSessionid(id);
userlist.add(newuser);
arg0.getServletContext().setAttribute("userlist", userlist);
}
}
}
在这个监听器中,我们只需要在请求初始化的时候,根据这个请求产生的SessionID判断,当前SessionID
所属的用户是否存在于服务器用户列表中,如果不存在,就新建一个用户(包含IP和SessionID信息),并
存入服务器用户列表。当然当第一个用户访问服务器时,用户列表是空的,为了防止出现空指针异常,我们
要在此时为服务器创建一个用户列表。
2.HttpSession监听器
public class Sessionlisten implements HttpSessionListener {
private int count;
public void sessionCreated(HttpSessionEvent arg0) {
// TODO Auto-generated method stub
System.out.println("-----新建会话-----");
count++;
arg0.getSession().getServletContext().setAttribute("usercount", count);
}
public void sessionDestroyed(HttpSessionEvent arg0) {
// TODO Auto-generated method stub
count--;
arg0.getSession().getServletContext().setAttribute("usercount", count);
System.out.println("-----会话销毁-----");
List<User> userlist = (List<User>) arg0.getSession()
.getServletContext().getAttribute("userlist");
String id = arg0.getSession().getId();
User user = (User) Getuserbyid.finduser(userlist, id);
userlist.remove(user);
arg0.getSession().getServletContext()
.setAttribute("userlist", userlist);
System.out.println("-------当前用户从服务器删除-----");
}
}
在Session对象监听器中,当一个会话创建时(有新用户访问),我们设置访问量加1,
当一个会话销毁时,表示一个用户掉线了,没有对服务器进行访问了,那么访问量减1,并且我们通过当
前会话ID获得这个用户,并且将这个用户从服务器用户列表删除。
下面是根据SessionID查询用户的功能类:
public class Getuserbyid {
public static Object finduser(List<User> userlist,String id){
if(userlist!=null){
for(User user:userlist){
if(user.getSessionid().equals(id)){
return user;
}
}
}
return null;
}
}
因为Session对象销毁时之前,这个Session对象所属的用户信心必定是早就存到上下文对象中了(服务器用
户列表),那么自然是存在的,我们就根据SessionID这个字段来遍历所有用户的SessionID属性,从而找到
这个用户,将其从列表删除。
注意:监听器创建之后要进行注册,不然是不能够起到监听作用的。还有就是不要忘记写一个页面来展示一下
存在于用户列表的所有用户信息。
<%
List<User> userlist=(List<User>)request.getServletContext().getAttribute("userlist");
%>
在线人数:<%=userlist.size() %><br> <%
if(userlist!=null){
for (User user :userlist){
%>
IP地址:<%=user.getIp() %>----sessionId:<%=user.getSessionid() %><br> <%
}}
%>
在线人数:2
IP地址:0:0:0:0:0:0:0:1----sessionId:18166147D09F290A25924ACF9753E19D 同一台电脑两个浏览器访问,IP自然相同
IP地址:0:0:0:0:0:0:0:1----sessionId:43C373670AF88EE244CBF7E6585F2285
完结撒花~~~~~~