监听器实现案例----自定义session扫描器和统计在线用户人数及用户信息

Posted feiwenstyle

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了监听器实现案例----自定义session扫描器和统计在线用户人数及用户信息相关的知识,希望对你有一定的参考价值。

一、案例一:自定义Session扫描器
1、案例说明
当一个Web应用创建的Session很多时,为了避免Session占用太多的内存,我们可以选择手动将这些内存中的session销毁,那么此时也可以借助监听器技术来实现。对于拿到 每个session 对象, 判断session的最后一次访问时间 与当前时间 的间隔是否超过 5 分钟, 如果超过就手动销毁

2、实现代码
SessionScanner:session对象的监听器

MyTimerTask:定时器timer的任务对象

 

SessionScanner监听器,使用servlet3.0新特性,使用注解@WebListener完成注册

(1)SessionScanner类
package sessionScanner;

import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
import java.util.Timer;

import javax.servlet.ServletContextEvent;
import javax.servlet.ServletContextListener;
import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpSession;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;

/*
* 自定义session扫描器的实现
* 手动的 自己去管理 session 对象, 如果某个session,5分钟没有被访问过,那么就销毁
*
*
* 1、监听session对象的修改时间,要设置监听器:HttpSessionListener(session产生和销毁时)
* 注意:定义一个容器,用来装session对象,在session产生时,添加到容器中;
* 在session销毁时,从容器中移除;由此进行管理
* 2、设置一个定时器(timer),每隔5分钟进行检查一次,看看哪些session是超过5分钟没有被访问过,
* 如果没有,则销毁
* 3、定时器timer有个任务对象,我们需要自己去创建这个任务对象:遍历检查session中超过5分钟的情况
* 而这个任务对象(TimerTask)是Runnable接口的实现类,所以只需实现Runnable的接口即可
* 4、定时器的启动需要进行设置,因此还要设置一个监听器:ServletContextListener;
* 目的:设置监听器,web应用启动时开始工作,然后每隔5分钟检查一次
*
*小结:
* 1. 增删频繁的时候, 使用 LinkedList 性能好
* 2. 如何将一个list 变为一个线程安全的list,使用Collections.synchronizedList
* 3. 定时器的使用 --- Timer 类
* 4. 遍历list集合的时候, 同时还要从list中去移除 元素, 避免 并发修改的异常(用ListIterator,而不是Iterator)
* 5. 如何实现两段不同的代码 互斥,锁的使用
*/

/*
*
* (1)实现HttpSessionListener监听器,实现两个方法
* public void sessionCreated(HttpSessionEvent se) {}
* public void sessionDestroyed(HttpSessionEvent se) {}
*
* (2)实现ServletContextListener监听器,实现两个方法
* public void contextInitialized(ServletContextEvent sce) {}
* public void contextDestroyed(ServletContextEvent sce) {}
*/

//监听器(观察者)
@WebListener
public class SessionScanner implements HttpSessionListener,
ServletContextListener {// 自定义session扫描器的实现

// 定义一个容器, 将 每次 创建的session 对象放到 容器中去
// private List<HttpSession> list=new LinkedList<HttpSession>();//线程不安全
private List<HttpSession> list = Collections
.synchronizedList(new LinkedList<HttpSession>());// 线程安全

public Object lock = new Object();// 定义一个锁,用于解决线程安全问题

// 主要解决:本对象的sessionCreated方法添加session,而MyTimerTask对象中方法销毁对象时,使用的是同一个session容器,
// 这样,对同一个容器做不同的操作,肯能产生线程安全问题,所以要定义锁:lock,去解决这个线程安全问题

@Override
// 事件对象(封装 session事件源 )
public void sessionCreated(HttpSessionEvent se) {
System.out.println("执行了,说明新创建了一个session对象。。。");

HttpSession session = se.getSession();// 获取事件对象
// 定义锁,解决线程安全问题
synchronized (lock) {
list.add(session);// 穿件的session放到容器中去
}

// long lastAccessedTime = session.getLastAccessedTime();//最后一次修改时间
}

@Override
public void sessionDestroyed(HttpSessionEvent se) {
System.out.println("执行了,说明销毁了一个session对象。。。");
}

@Override
public void contextInitialized(ServletContextEvent sce) {

System.out.println("contextInitialized..............");
// 定义一个定时器,并且在web应用启动时开始工作
Timer timer = new Timer();

// 安排指定的任务从指定的延迟后开始进行重复的固定延迟执行
// task:安排的任务
// delay:举例开始的指定的延时时间
// period:重复时间

//立刻 启动 定时器, 每隔 5 分钟 重复 执行
timer.schedule(new MyTimerTask(list,lock), 0, 1000 * 60 * 5);// 1000毫秒*60*5=5分钟
}

@Override
public void contextDestroyed(ServletContextEvent sce) {

System.out.println("contextDestroyed......");
}

}
(2)MyTimerTask类
package sessionScanner;

import java.util.List;
import java.util.ListIterator;
import java.util.TimerTask;

import javax.servlet.http.HttpSession;

//任务对象
class MyTimerTask extends TimerTask {

private List<HttpSession> list;// session容器
private Object lock;//锁,从SessionScanner中 传递而来

public MyTimerTask(List<HttpSession> list,Object lock) {
this.list = list;// 获取session容器
this.lock=lock;//锁,从SessionScanner中 传递而来
}

@Override
public void run() {
// 定义锁,解决线程安全问题
synchronized (lock) {
// 遍历session容器
ListIterator<HttpSession> it = list.listIterator();
while (it.hasNext()) {
HttpSession session = it.next();
// 遍历 list , 拿到 每个session 对象, 判断session的最后一次访问时间 与当前时间 的间隔是否超过 5 分钟, 如果超过就手动销毁

if (session.getLastAccessedTime() - 1000 * 60 * 5 > 0) {
session.invalidate();// 销毁session
list.remove(session);// 从session容器中销毁session对象
}
}
}
}

}
二、案例二:统计在线用户人数及用户信息
1、案例说明
 统计在线用户人数及用户信息

  1、在线用户的数量

  2、在线用户的信息:sessionId,ip地址,上一次访问时间

  做法:

  1、统计用户数量,通过HttpSessionListener完成             并且定义number来存储数量(通过ServletContext设置,获取)

2、统计用户信息,通过ServletRequestListener完成    

并且定义List容器来存储所有用户信息(通过ServletContext设置,获取)

  备注:使用时,需要自己打开不同的浏览器,才会有测试效果

2、实现代码
User:用户的数据封装对象

Count:统计在线用户数量

CountInfo:统计在线用户信息

Util:工具类,用于判断所有用户中,是否存在当前用户的访问信息

index.jsp页面:在线用户人数和在线用户信息显示

(1)User
package countUser;

/*
* 统计在线用户人数及用户信息
* 1、在线用户的数量
* 2、在线用户的信息:sessionId,ip地址,上一次访问时间
*
* 做法:
* 1、统计用户数量,通过HttpSessionListener完成 --定义number来存储数量(通过ServletContext设置,获取)
* 2、统计用户信息,通过ServletRequestListener完成 --定义List容器来存储所有用户信息(通过ServletContext设置,获取)
*
* 备注:使用时,需要自己打开不同的浏览器,才会有测试效果
*/
public class User {

private String sessionId;
private String ip;
private String firstTime;
public String getSessionId() {
return sessionId;
}
public void setSessionId(String sessionId) {
this.sessionId = sessionId;
}
public String getIp() {
return ip;
}
public void setIp(String ip) {
this.ip = ip;
}
public String getFirstTime() {
return firstTime;
}
public void setFirstTime(String firstTime) {
this.firstTime = firstTime;
}

}
(2)Count
package countUser;

import java.util.ArrayList;

import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;

//Servlet3.0 新特性
//1、统计用户数量,通过HttpSessionListener完成 --定义number来存储数量(通过ServletContext设置,获取)

@WebListener
public class Count implements HttpSessionListener {

private int num=0;//统计用户在线人数
@Override
public void sessionCreated(HttpSessionEvent se) {

num++;
//设置到ServletContext域中
se.getSession().getServletContext().setAttribute("num",num);
System.out.println(" add....");
}

@Override
public void sessionDestroyed(HttpSessionEvent se) {

num--;
//设置到ServletContext域中
se.getSession().getServletContext().setAttribute("num",num);
System.out.println(" remove....");

//注意,此处还要进行设置
//从session列表中移除session
ArrayList<User> list=null;
list=(ArrayList<User>) se.getSession().getServletContext().getAttribute("list");

if(Util.getByUserId(list, se.getSession().getId())!=null){
list.remove(Util.getByUserId(list, se.getSession().getId()));
}
}

}
(3)CountInfo
package countUser;

import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;

import javax.servlet.ServletRequestEvent;
import javax.servlet.ServletRequestListener;
import javax.servlet.annotation.WebListener;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
//2、统计用户信息,通过ServletRequestListener完成 --定义List容器来存储所有用户信息(通过ServletContext设置,获取)

@WebListener
public class CountImfo implements ServletRequestListener {

private ArrayList<User> list;//存储访问用户的信息



@Override
public void requestInitialized(ServletRequestEvent sre) {
//获得ServletContext域中的list
list=(ArrayList<User>) sre.getServletContext().getAttribute("list");

// if(list.isEmpty())
if(list==null){
list=new ArrayList<User>();
}
//为了获得session对象,进行强制类型转换
HttpServletRequest request=(HttpServletRequest)sre.getServletRequest();
HttpSession session = request.getSession();
String sessionId = session.getId();//sessionId

//session的列表中没有当前的sessionId,即:以前的所有访问用户中,没有当前的访问用户,所以把当前的访问用户信息加入
if(Util.getByUserId(list,sessionId)==null){
User user=new User();
user.setSessionId(sessionId);
user.setIp(request.getRemoteAddr());
//设置时间
user.setFirstTime(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
// user.setFirstTime(System.currentTimeMillis()+"");
list.add(user);
}
// request.getServletContext().setAttribute("list", list);//保存所有用户信息到list中
sre.getServletContext().setAttribute("list", list);//保存所有用户信息到list中
}

@Override
public void requestDestroyed(ServletRequestEvent sre) {

}

}
(4)Util           
package countUser;

import java.util.ArrayList;

public class Util {

//用于判断所有用户中,是否存在当前用户的访问信息
public static Object getByUserId(ArrayList<User> list, String sessionId) {

// if(!list.isEmpty()){
for(int i=0;i<list.size();i++){
User user = list.get(i);
if(user.getSessionId().equals(sessionId)){
return user;//所有用户中有,已经存在当前用户的访问信息
}
}
// }
return null;//所有用户中有,不存在当前用户的访问信息
}
}
(5)index.jsp页面
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
</head>

<body>
当前在线用户人数:${num }<br/>
<!-- 当前用户在线人数: ${num}<br /> -->


所有访问者的信息列表<br/>
<c:if test="${ empty list}">
当前不存在访问者
</c:if>

<c:if test="${ not empty list}">
<c:forEach var="user" items="${list }">
sessionId:${user.sessionId }  
ip:${user.ip }  
firstTime:${user.firstTime }  
<br/>
</c:forEach>
</c:if>


<!--
<br/>遍历方式二:<br/>
<%
ArrayList<countUser.User> userList = (ArrayList<countUser.User>)request.getServletContext().getAttribute("list");
if(userList!=null){
for(int i = 0 ; i < userList.size() ; i++){
countUser.User user = userList.get(i);
%>
IP:<%=user.getIp() %>,FirstTime:<%=user.getFirstTime() %>,SessionId:<%=user.getSessionId() %> <br/>
<%}} %>


-->
</body>
</html>
3、实现结果

以上是关于监听器实现案例----自定义session扫描器和统计在线用户人数及用户信息的主要内容,如果未能解决你的问题,请参考以下文章

重学SpringBoot系列之生命周期内的拦截过滤与监听

自定义session扫描器精确控制session销毁时间--学习笔记

简单版 -- 通过Session统计用户对某一JSP页面的在线人数

微信小程序-- 自定义组件 - 数据监听器 - 案例 (三十五)

Flask中的session ,自定义实现 session机制, 和 flask-session组件

通过django中间件和python魔法方法实现自定义session(通过文件存储session)