问题描述 : 在做网站次数统计的时候,需要用到上下文对象,最终在上下文监听中发现上下文对象创建两次,销毁两次,导致数据库中网站访问次数统计更新失败。
原因 : 我们更改了项目的访问路径,就会导致初始化两个上下文对象,同时销毁也是销毁两个,在上下文监听里边打印上下文对象,可以发现这个现象。
查看上下文监听中的日志,发现会创建的两个上下文对象,一个是stu,一个是JavaEE-Study,而且这两个上下文对象内存地址不一致,也就是说,不是同一个。
上下文监听核心代码
public class MyContextListener implements ServletContextListener{
public void contextInitialized(ServletContextEvent sce) {
System.out.println("上下文对象被创建");
ServletContext context = sce.getServletContext();
System.out.println("context = " + context);
String contextPath = context.getContextPath();
System.out.println("contextPath = " + contextPath);
// 查询数据库,获取之前的网站访问次数,代码略,假设查询到的值是999
// 存储数据只会放在了stu上下文中,JavaEE-Study上下文中不会存储进去
context.setAttribute("webCount", 999);
}
public void contextDestroyed(ServletContextEvent sce) {
System.out.println("上下文对象被销毁");
ServletContext context = sce.getServletContext();
/**
* 随着程序的运行,网站访问次数为1024次
* 在销毁的时候,会触发两次该方法
* 第一次要销毁stu上下文,从stu上下文中获取到值1024,更新到数据库中
* 第二次要销毁JavaEE-Study这个上下文对象,
* 但是从JavaEE-Study上下文中获取到的不是1024这个值,而是999
* 那么999就会更新到数据库,覆盖掉第一次更新的1024,最终导致更新失败
*/
Integer webCount = (Integer) context.getAttribute("webCount");
// 更新到数据库,代码 略
}
}
启动项目,查看日志 : 有两个不一样的上下文对象
解决方案
在上下文监听中,销毁上下文对象的方法中,获取上下文名称,根据上下文名称进行判断,在正确的上下文名称中进行更新网站访问次数的代码。
public void contextDestroyed(ServletContextEvent sce) {
System.out.println("上下文对象被销毁");
ServletContext context = sce.getServletContext();
Integer webCount = (Integer) context.getAttribute("webCount");
String contextPath = context.getContextPath();
if ("stu".equals(context)) {
// 更新到数据库,代码 略
}
}