假设我有一个正在运行的基于Java的Web应用程序,具有0个或多个与之关联的有效ServletContextListener对象。 我想要一种访问有效contextInitialized()对象的当前列表的方法。 我以为可以实现HttpSession,并使用它附加到存储在应用程序作用域属性中的会话ID值列表中,但是当会话无效时,我想立即更新列表,谁知道 还有什么。
在开始烘焙自己的解决方案之前,我想我应该问一个问题:
Servlet API是否提供某些方法来访问非无效会话对象的完整列表?
我正在使用Tomcat 6.x作为我的Web应用程序容器,以及MyFaces 1.2.x(JSF)库。
解
我遵循的方法类似于BalusC在这些现有问题中讨论的方法:
如何轻松实现“谁是或Java应用程序中的“在线”?
JSF:如何使他使用两次登录时的用户会话相同的凭证
我通过ServletContextListener类进行了修改,以实现contextInitialized()。发生绑定事件时,该对象将自己添加或删除所有HttpSession对象的集合。
@Override
public void valueBound(HttpSessionBindingEvent event) {
// Get my custom application-scoped attribute
ApplicationData applicationData = getApplicationData();
// Get the set of all SessionData objects and add myself to it
Set activeSessions = applicationData.getActiveSessions();
if (!activeSessions.contains(this)) {
activeSessions.add(this);
}
}
@Override
public void valueUnbound(HttpSessionBindingEvent event) {
HttpSession session = event.getSession();
ApplicationData applicationData = getApplicationData();
Set activeSessions = applicationData.getActiveSessions();
if (activeSessions.contains(this)) {
activeSessions.remove(this);
}
}
继续困扰我的是,重新启动Tomcat时会发生什么。 除非已将Tomcat正确配置为不将会话序列化到磁盘,否则它将这样做。 当Tomcat重新启动时,ServletContextListener对象(以及contextInitialized()对象及其对象)将反序列化,并使会话再次有效。 但是,序列化/反序列化完全避开了HttpSession侦听器事件,因此重新启动后,我没有机会优雅地将对SessionData的反序列化引用放回我的托管对象集中。
我对客户组织中的Tomcat的生产配置没有任何控制权,因此我无法假设它将按我期望的方式完成。
我的解决方法是将ServletContextListener的创建时间与收到请求时的应用程序启动时间进行比较。 如果会话是在应用程序启动时间之前创建的,那么我会致电contextInitialized(),并将用户发送到错误/警告页面,并说明发生的情况。
我通过实现ServletContextListener并从我的侦听器的contextInitialized()方法内部将当前时间存储在应用程序范围内的对象中,来获取应用程序的启动时间。