在Struts2下使用[拦截器]配合[session监听器]实现在线会员统计与防止重复登录

需求:

1.管理员后台可显示当前在线访客数以及在线会员数(在线会员要求列出详细清单).

2.某个客户端非法退出(直接关闭浏览器或断电啥的)后,接着他重新登录系统.在线统计上要做到保持登录统计的准确性与完整性.

3.防止多个客户端同时使用一个帐户异地登录(后登录者踢掉先登录者,并通知先登录者被迫下线).

 

 

1.监听器SessionListener (web.xml的代码略)

Java代码 复制代码  收藏代码
  1. /**  
  2.      * 创建session  
  3.      */  
  4.     public void sessionCreated(HttpSessionEvent se) {   
  5.         OnlineStatistics.increase();   
  6.     }   
  7.   
  8.     /**  
  9.      * 服务器销毁session对象  
  10.      */  
  11.     public void sessionDestroyed(HttpSessionEvent se) {   
  12.         OnlineStatistics.decrease();   
  13.     }   
  14.   
  15.     /**  
  16.      * 用户登录时  
  17.      */  
  18.     public void attributeAdded(HttpSessionBindingEvent se) {   
  19.         if ("memberInfo".equals(se.getName())) {   
  20.             MemberInfo memberInfo = (MemberInfo) se.getValue();   
  21.             OnlineStatistics.addAttr(memberInfo);   
  22.             se.getSession().setMaxInactiveInterval(60 * 20);// 失效时间SEC * MINS   
  23.         }   
  24.     }   
  25.   
  26.     /**  
  27.      * 用户退出登录(销毁session属性时)  
  28.      */  
  29.     public void attributeRemoved(HttpSessionBindingEvent se) {   
  30.         if ("memberInfo".equals(se.getName())) {   
  31.             MemberInfo memberInfo = (MemberInfo) se.getValue();   
  32.             OnlineStatistics.removeAttr(memberInfo);   
  33.             memberInfo = null;   
  34.         }   
  35.     }   
  36.   
  37.     /**  
  38.      * 待实现  
  39.      */  
  40.     public void attributeReplaced(HttpSessionBindingEvent se) {   
  41. //to be extended   
  42.     }  
/**
	 * 创建session
	 */
	public void sessionCreated(HttpSessionEvent se) {
		OnlineStatistics.increase();
	}

	/**
	 * 服务器销毁session对象
	 */
	public void sessionDestroyed(HttpSessionEvent se) {
		OnlineStatistics.decrease();
	}

	/**
	 * 用户登录时
	 */
	public void attributeAdded(HttpSessionBindingEvent se) {
		if ("memberInfo".equals(se.getName())) {
			MemberInfo memberInfo = (MemberInfo) se.getValue();
			OnlineStatistics.addAttr(memberInfo);
			se.getSession().setMaxInactiveInterval(60 * 20);// 失效时间SEC * MINS
		}
	}

	/**
	 * 用户退出登录(销毁session属性时)
	 */
	public void attributeRemoved(HttpSessionBindingEvent se) {
		if ("memberInfo".equals(se.getName())) {
			MemberInfo memberInfo = (MemberInfo) se.getValue();
			OnlineStatistics.removeAttr(memberInfo);
			memberInfo = null;
		}
	}

	/**
	 * 待实现
	 */
	public void attributeReplaced(HttpSessionBindingEvent se) {
//to be extended
	}

 

 2. 统计类OnlineStatistics

Java代码 复制代码  收藏代码
  1. /**  
  2.  * @author Arantam  
  3.  */  
  4. package com.farago.util;   
  5.   
  6. import java.util.Date;   
  7. import java.util.HashSet;   
  8. import java.util.Iterator;   
  9. import java.util.Set;   
  10.   
  11. import com.farago.vo.MemberInfo;   
  12.   
  13. /**  
  14.  * @author Arantam  
  15.  *   
  16.  */  
  17. @SuppressWarnings("unchecked")   
  18. public class OnlineStatistics {   
  19.     private static int count = 0;   
  20.     private static Set onlineMemberList = new HashSet();   
  21.   
  22.     /**  
  23.      * 统计在线总数(递增)  
  24.      */  
  25.     public static void increase() {   
  26.         count++;   
  27.     }   
  28.   
  29.     /**  
  30.      * 统计在线人数(递减)  
  31.      */  
  32.     public static void decrease() {   
  33.         count--;   
  34.     }   
  35.   
  36.     /**  
  37.      * 统计在线会员(添加)  
  38.      *   
  39.      * @param memberInfo  
  40.      */  
  41.     public static void addAttr(MemberInfo memberInfo) {   
  42.         // 如果在线会员列表不为空,则迭代该列表,检查当前登录的会员id与列表中的记录id是否一致.   
  43.         Iterator itr = onlineMemberList.iterator();   
  44.         while (itr.hasNext()) {   
  45.             // 先判断该会员是否已有session信息保存在统计对象中.若有,把旧的挤掉,再放入新的.   
  46.             MemberInfo temp = (MemberInfo) itr.next();   
  47.             String memberName = temp.getName();// 保存在统计类中的用户名   
  48.             if (memberInfo.getName().equals(memberName)) {   
  49.                 itr.remove();   
  50.             }   
  51.         }   
  52.         onlineMemberList.add(memberInfo);   
  53.     }   
  54.   
  55.     /**  
  56.      * 统计在线会员(清除)  
  57.      *   
  58.      * @param memberInfo  
  59.      */  
  60.     public static void removeAttr(MemberInfo memberInfo) {   
  61.         onlineMemberList.remove(memberInfo);   
  62.     }   
  63.   
  64.     /**  
  65.      * 获取在线访客的数量  
  66.      *   
  67.      * @return the count  
  68.      */  
  69.     public static int getOnlineVisitors() {   
  70.         return count - 1;// 除去管理员本身   
  71.     }   
  72.   
  73.     /**  
  74.      * 获取在线会员的列表  
  75.      *   
  76.      * @return the onlineMemberList  
  77.      */  
  78.     public static Set getOnlineMemberList() {   
  79.         return onlineMemberList;   
  80.     }   
  81.   
  82.     /**  
  83.      * 根据登录时间判断登录次序的先后  
  84.      *   
  85.      * @param memberInfo  
  86.      * @return  
  87.      */  
  88.     public static boolean isOld(MemberInfo memberInfo) {   
  89.         Iterator itr = onlineMemberList.iterator();   
  90.         while (itr.hasNext()) {   
  91.             MemberInfo temp = (MemberInfo) itr.next();   
  92.             String memberName = temp.getName();// 保存在统计类中的用户名   
  93.             Date loginTime = temp.getLoginTime();// 保存在统计类中的用户登录时间   
  94.             if (memberInfo.getName().equals(memberName) && memberInfo.getLoginTime().before(loginTime)) {   
  95.                 // 如果当前用户的登录时间比统计对象里保存的同名用户的登录时间要早,则返回true   
  96.                 return true;   
  97.             }   
  98.         }   
  99.         return false;   
  100.     }   
  101.   
  102. }  
/**
 * @author Arantam
 */
package com.farago.util;

import java.util.Date;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;

import com.farago.vo.MemberInfo;

/**
 * @author Arantam
 * 
 */
@SuppressWarnings("unchecked")
public class OnlineStatistics {
	private static int count = 0;
	private static Set onlineMemberList = new HashSet();

	/**
	 * 统计在线总数(递增)
	 */
	public static void increase() {
		count++;
	}

	/**
	 * 统计在线人数(递减)
	 */
	public static void decrease() {
		count--;
	}

	/**
	 * 统计在线会员(添加)
	 * 
	 * @param memberInfo
	 */
	public static void addAttr(MemberInfo memberInfo) {
		// 如果在线会员列表不为空,则迭代该列表,检查当前登录的会员id与列表中的记录id是否一致.
		Iterator itr = onlineMemberList.iterator();
		while (itr.hasNext()) {
			// 先判断该会员是否已有session信息保存在统计对象中.若有,把旧的挤掉,再放入新的.
			MemberInfo temp = (MemberInfo) itr.next();
			String memberName = temp.getName();// 保存在统计类中的用户名
			if (memberInfo.getName().equals(memberName)) {
				itr.remove();
			}
		}
		onlineMemberList.add(memberInfo);
	}

	/**
	 * 统计在线会员(清除)
	 * 
	 * @param memberInfo
	 */
	public static void removeAttr(MemberInfo memberInfo) {
		onlineMemberList.remove(memberInfo);
	}

	/**
	 * 获取在线访客的数量
	 * 
	 * @return the count
	 */
	public static int getOnlineVisitors() {
		return count - 1;// 除去管理员本身
	}

	/**
	 * 获取在线会员的列表
	 * 
	 * @return the onlineMemberList
	 */
	public static Set getOnlineMemberList() {
		return onlineMemberList;
	}

	/**
	 * 根据登录时间判断登录次序的先后
	 * 
	 * @param memberInfo
	 * @return
	 */
	public static boolean isOld(MemberInfo memberInfo) {
		Iterator itr = onlineMemberList.iterator();
		while (itr.hasNext()) {
			MemberInfo temp = (MemberInfo) itr.next();
			String memberName = temp.getName();// 保存在统计类中的用户名
			Date loginTime = temp.getLoginTime();// 保存在统计类中的用户登录时间
			if (memberInfo.getName().equals(memberName) && memberInfo.getLoginTime().before(loginTime)) {
				// 如果当前用户的登录时间比统计对象里保存的同名用户的登录时间要早,则返回true
				return true;
			}
		}
		return false;
	}

}

 

 

3. 拦截器代码片断

Java代码 复制代码  收藏代码
  1. Map sessionMap = invocation.getInvocationContext().getSession();   
  2.         if (sessionMap.get("memberInfo") != null) {   
  3.             MemberInfo memberInfo = (MemberInfo) sessionMap.get("memberInfo");   
  4.   
  5.             /**  
  6.              *   
  7.              */  
  8.             if (OnlineStatistics.isOld(memberInfo)) {   
  9.                 sessionMap.remove("memberInfo");// 清除session   
  10.                 faragoMessage = "对不起,您的帐号[" + memberInfo.getName() + "]已在别的地方登录,您已被迫退出。若有疑问请联系管理员,谢谢!";   
  11.                 ctx.put("faragoMessage", faragoMessage);   
  12.                 return Action.ERROR;   
  13.             }   
  14. ......   
  15. ...  
Map sessionMap = invocation.getInvocationContext().getSession();
		if (sessionMap.get("memberInfo") != null) {
			MemberInfo memberInfo = (MemberInfo) sessionMap.get("memberInfo");

			/**
			 * 
			 */
			if (OnlineStatistics.isOld(memberInfo)) {
				sessionMap.remove("memberInfo");// 清除session
				faragoMessage = "对不起,您的帐号[" + memberInfo.getName() + "]已在别的地方登录,您已被迫退出。若有疑问请联系管理员,谢谢!";
				ctx.put("faragoMessage", faragoMessage);
				return Action.ERROR;
			}
......
...

 

 

 

如果想在前台也显示当前在线人数,可以在sessionCreated和sessionDestroyed方法中处理,把统计数字放入ServletContext里面.然后使用S2的标签直接显示.这样来得方便.呵呵

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值