ServletContextListener使用详解

ServletContextListener使用详解在 ServletAPI 中有一个 ServletContextListener 接口,它能够监听 ServletContext 对象的生命周期,实际上就是监听 Web 应用的生命周期。当Servlet 容器启动或终止Web 应用时,会触发ServletContextEvent 事件,该事件由ServletContextListener 来处理。在 ServletContextListene

大家好,又见面了,我是你们的朋友全栈君。

在 Servlet API 中有一个 ServletContextListener 接口,它能够监听 ServletContext 对象的生命周期,实际上就是监听 Web 应用的生命周期。

当Servlet 容器启动或终止Web 应用时,会触发ServletContextEvent 事件,该事件由ServletContextListener 来处理。在 ServletContextListener 接口中定义了处理ServletContextEvent 事件的两个方法。

Java代码  
收藏代码

  1. /** 
  2.  * 当Servlet 容器启动Web 应用时调用该方法。在调用完该方法之后,容器再对Filter 初始化, 
  3.  * 并且对那些在Web 应用启动时就需要被初始化的Servlet 进行初始化。 
  4.  */  
  5. contextInitialized(ServletContextEvent sce)   
  6.   
  7.   
  8. /** 
  9.  * 当Servlet 容器终止Web 应用时调用该方法。在调用该方法之前,容器会先销毁所有的Servlet 和Filter 过滤器。 
  10.  */  
  11. contextDestroyed(ServletContextEvent sce)  

 

 下面通过两个具体的例子来介绍 ServletContextListener 的用法。

 

例一:在服务启动时,将数据库中的数据加载进内存,并将其赋值给一个属性名,其它的 Servlet 就可以通过 getAttribute 进行属性值的访问。

 

有如下两个步骤:

 

1 : ServletContext 对象是一个为整个 web 应用提供共享的内存,任何请求都可以访问里面的内容  

 

2 :如何实现在服务启动的时候就动态的加入到里面的内容:我们需要做的有:  

1 ) 实现 servletContextListerner 接口 并将要共享的通过 setAttribute ( name,data )方法提交到内存中去   ;

2 )应用项目通过 getAttribute(name) 将数据取到 。

 

Java代码  
收藏代码

  1. public class ServletContextLTest implements ServletContextListener{   
  2.   
  3.     // 实现其中的销毁函数  
  4.       
  5.     public void contextDestroyed(ServletContextEvent sce) {   
  6.   
  7.         System.out.println(“this is last destroyeed”);      
  8.   
  9.     }   
  10.   
  11.     // 实现其中的初始化函数,当有事件发生时即触发  
  12.   
  13.     public void contextInitialized(ServletContextEvent sce) {   
  14.   
  15.         ServletContext sct=sce.getServletContext();   
  16.   
  17.         Map<Integer,String> depts=new HashMap<Integer,String>();   
  18.   
  19.         Connection connection=null;   
  20.   
  21.         PreparedStatement pstm=null;   
  22.   
  23.         ResultSet rs=null;   
  24.   
  25.            
  26.   
  27.         try{   
  28.   
  29.             connection=ConnectTool.getConnection();   
  30.   
  31.             String sql=”select deptNo,dname from dept”;   
  32.   
  33.             pstm=connection.prepareStatement(sql);   
  34.   
  35.             rs=pstm.executeQuery();   
  36.   
  37.             while(rs.next()){   
  38.   
  39.                 depts.put(rs.getInt(1), rs.getString(2));   
  40.   
  41.             }   
  42.   
  43.             // 将所取到的值存放到一个属性键值对中  
  44.   
  45.             sct.setAttribute(“dept”, depts);   
  46.   
  47.             System.out.println(“======listener test is beginning=========”);   
  48.   
  49.         }catch(Exception e){   
  50.   
  51.             e.printStackTrace();   
  52.   
  53.         }finally{   
  54.   
  55.             ConnectTool.releasersc(rs, pstm, connection);   
  56.   
  57.         }   
  58.   
  59.     }   
  60.   
  61. }  

 

在完成上述编码后,仍需在 web.xml 中进行如下配置,以使得该监听器可以起作用。

 

Xml代码  
收藏代码

  1. <listener>   
  2.   
  3.    <listener-class>ServletContextTest.ServletContextLTest</listener-class>   
  4.   
  5. </listener>    

 

在完成上述配置后, web 服务器在启动时,会直接加载该监听器,通过以下的应用程序就可以进行数据的访问。

 

Java代码  
收藏代码

  1. public class CreateEmployee extends HttpServlet{   
  2.   
  3.     @Override   
  4.   
  5.     protected void service(HttpServletRequest request, HttpServletResponse response)   
  6.   
  7.             throws ServletException, IOException {   
  8.   
  9.         ServletContext sct=getServletConfig().getServletContext();   
  10.   
  11.         // 从上下文环境中通过属性名获取属性值  
  12.   
  13.         Map<Integer,String> dept=(Map<Integer,String>)sct.getAttribute(“dept”);   
  14.   
  15.         Set<Integer> key=dept.keySet();   
  16.   
  17.         response.setContentType(“text/html;charset=utf-8”);   
  18.   
  19.         PrintWriter out=response.getWriter();   
  20.   
  21.         out.println(“<html>”);   
  22.   
  23.         out.println(“<body>”);   
  24.   
  25.         out.println(“<form action=’/register’ action=’post’>”);   
  26.   
  27.         out.println(“<table alignb=’center’>”);   
  28.   
  29.         out.println(“<tr>”);   
  30.   
  31.         out.println(“<td>”);   
  32.   
  33.         out.println(“username:”);   
  34.   
  35.         out.println(“</td>”);   
  36.   
  37.         out.println(“<td>”);   
  38.   
  39.         out.println(“<input type=’text’ name=’username'”);   
  40.   
  41.         out.println(“</tr>”);   
  42.   
  43.         out.println(“<tr>”);   
  44.   
  45.         out.println(“<td>”);   
  46.   
  47.         out.println(“city:”);   
  48.   
  49.         out.println(“</td>”);   
  50.   
  51.         out.println(“<td>”);   
  52.   
  53.         out.println(“<select name=’dept'”);   
  54.   
  55.         for(Integer i:key){   
  56.   
  57.             out.println(“<option value='”+i+”‘>”+dept.get(i)+”</option>”);   
  58.   
  59.         }   
  60.   
  61.         out.println(“</select>”);   
  62.   
  63.         out.println(“</td>”);   
  64.   
  65.         out.println(“<tr>”);   
  66.   
  67.         out.println(“</table>”);   
  68.   
  69.         out.println(“</form>”);   
  70.   
  71.         out.println(“</body>”);   
  72.   
  73.         out.println(“</html>”);   
  74.   
  75.         out.flush();   
  76.   
  77.     }   
  78.   
  79. }  

 

例二:书写一个类用于统计当Web 应用启动后,网页被客户端访问的次数。如果重新启动Web 应用,计数器不会重新从1 开始统计访问次数,而是从上次统计的结果上进行累加。

 

在实际应用中,往往需要统计自Web 应用被发布后网页被客户端访问的次数,这就要求当Web 应用被终止时,计数器的数值被永久存储在一个文件中或者数据库中,等到Web 应用重新启动时,先从文件或数据库中读取计数器的初始值,然后在此基础上继续计数。

 

向文件中写入或读取计数器的数值的功能可以由自定义的 MyServletContextListener 类来完成,它具有以下功能:

 

1 、在 Web 应用启动时从文件中读取计数器的数值,并把表示计数器的 Counter 对象存放到 Web应用范围内。存放计数器的文件的路径为helloapp/count/count.txt 。

2 、在Web 应用终止时把Web 应用范围内的计数器的数值保存到count.txt 文件中。

Java代码  
收藏代码

  1. public class MyServletContextListener implements ServletContextListener{  
  2.   
  3.   public void contextInitialized(ServletContextEvent sce){  
  4.   
  5.     System.out.println(“helloapp application is Initialized.”);  
  6.   
  7.     // 获取 ServletContext 对象  
  8.   
  9.     ServletContext context=sce.getServletContext();  
  10.   
  11.     try{  
  12.   
  13.        // 从文件中读取计数器的数值  
  14.   
  15.        BufferedReader reader=new BufferedReader(  
  16.   
  17.            new InputStreamReader(context.  
  18.   
  19.            getResourceAsStream(“/count/count.txt”)));  
  20.   
  21.        int count=Integer.parseInt(reader.readLine());  
  22.   
  23.        reader.close();  
  24.   
  25.        // 创建计数器对象  
  26.   
  27.        Counter counter=new Counter(count);  
  28.   
  29.        // 把计数器对象保存到 Web 应用范围  
  30.   
  31.        context.setAttribute(“counter”,counter);  
  32.   
  33.        } catch(IOException e) {  
  34.   
  35.           e.printStackTrace();  
  36.   
  37.        }  
  38.   
  39.    }  
  40.   
  41.    public void contextDestroyed(ServletContextEvent sce){  
  42.   
  43.        System.out.println(“helloapp application is Destroyed.”);  
  44.   
  45.        // 获取 ServletContext 对象  
  46.   
  47.        ServletContext context=sce.getServletContext();  
  48.   
  49.        // 从 Web 应用范围获得计数器对象  
  50.   
  51.        Counter counter=(Counter)context.getAttribute(“counter”);  
  52.   
  53.        if(counter!=null){  
  54.   
  55.        try{  
  56.   
  57.           // 把计数器的数值写到 count.txt 文件中  
  58.   
  59.           String filepath=context.getRealPath(“/count”);  
  60.   
  61.           filepath=filepath+”/count.txt”;  
  62.   
  63.           PrintWriter pw=new PrintWriter(filepath);  
  64.   
  65.           pw.println(counter.getCount());  
  66.   
  67.           pw.close();  
  68.   
  69.          } catch(IOException e) {  
  70.   
  71.              e.printStackTrace();  
  72.   
  73.          }  
  74.   
  75.      }  
  76.   
  77.    }  
  78.   
  79. }   

 

将用户自定义的 MyServletContextListener 监听器在 Servlet 容器进行注册, Servlet 容器会在启动或终止 Web 应用时,会调用该监听器的相关方法。在 web.xml 文件中, <listener> 元素用于向容器注册监听器:

Xml代码  
收藏代码

  1. <listener>  
  2. <listener-class> ServletContextTest .MyServletContextListener<listener-class />  
  3. </listener>   

 

通过上述两个例子,即可以非常清楚的了解到 ServletContextListener 接口的使用方法及技巧。

 

在Container 加载Web 应用程序时(例如启动 Container 之后),会呼叫contextInitialized() ,而当容器移除Web 应用程序时,会呼叫contextDestroyed () 方法。

 

通过 Tomcat 控制台的打印结果的先后顺序,会发现当 Web 应用启动时,Servlet 容器先调用contextInitialized() 方法,再调用lifeInit 的init() 方法;

当Web 应用终止时,Servlet 容器先调用lifeInit 的destroy() 方法,再调用contextDestroyed() 方法。

 

由此可见,在Web 应用的生命周期中,ServletContext 对象最早被创建,最晚被销毁。

 

原文地址:http://blog.csdn.net/zhaozheng7758/archive/2010/12/28/6103700.aspx


版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/130593.html原文链接:https://javaforall.net

(0)
上一篇 2022年4月29日 上午9:33
下一篇 2022年4月29日 上午9:33


相关推荐

  • pcl点云合并_pcl点云重建

    pcl点云合并_pcl点云重建本节记录下点云聚类方法1.欧式聚类分割方法//为提取点云时使用的搜素对象利用输入点云cloud_filtered创建Kd树对象tree。pcl::search::KdTree::Ptrtree(newpcl::search::KdTree);tree-&amp;amp;gt;setInputCloud(cloud_filtered);//创建点云索引向量,用于存储实际的点云信息首先创…

    2022年10月21日
    3
  • linux 打包

    linux 打包zipzip 压缩后文件名源文件 zip r 压缩后目录名原目录 zip rwiki zipwiki unzip 解压 d 指定解压路径源文件 unzip dwikiwiki zip tartar cvf 打包名打包源文件 可以多个文件空格隔开 c 创建 v 可视 f 文件名 tar xvf 解包名 tar tf 查看包内文件打包 tar czvf gz 格式的 gzip 打包压缩包 tar cjvf bz2 格式的 bzip2 打包压缩包 t

    2026年3月19日
    2
  • C# 多线程使用lamda表达式编程

    C# 多线程使用lamda表达式编程C#多线程的实现方式使用的thread类1、最容易实现方式:Threadt=newThread(newThreadStart(ThreadFunction));publicvoidThreadFunction(){Console.WriteLine("我是线程!");}2、lamda表达式实现:Threadt=newThread(()=&gt;Co…

    2022年5月29日
    44
  • 你对贝叶斯统计都有怎样的理解?

    你对贝叶斯统计都有怎样的理解?作者:王冲链接:https://www.zhihu.com/question/21134457/answer/40753337来源:知乎著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。谢邀。Bayesian学派说概率是一个人对于一件事的信念强度,概率是主观的。而频率派是说概率是客观的。所有能用客观概率假设能解的题,用主观概率假设也都能解,答案一样。对

    2022年6月2日
    47
  • 【Redis】Redis配置文件详解

    【Redis】Redis配置文件详解一、存放位置linux下一定要养成备份配置文件的习惯。我是将配置文件拷贝至/myredis目录下进行操作的;二、Units单位这个在配置文件开始位置1.配置大小单位,开头定义了一些基本的度量单位,只支持bytes,不支持bit;2.对大小写不敏感。三、INCLUDES1.和Struts2配置文件类似,可以通过includes包含,redis.c…

    2022年6月7日
    55
  • 自锁电子开关电路_继电器自锁

    自锁电子开关电路_继电器自锁电路原理分析:电路上电后,由于IRF5305(P型MOSFET)栅极经R1上拉,处于高电位,IRF5305处于截止状态,当按下轻触开关S1,5V电源经R1,R2,R3,C1与R5分压得到一个能使Q2导通的电压,Q2一导通,Q2集电极电压拉低,IRF5305栅极电压也变低,IRF5305导通,电压经R4,R5分压…

    2026年1月31日
    4

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号