在Servlet API中有一个ServletContextListener接口,它能够监听ServletContext对象的生命周期,实际上就是监听Web应用的生命周期。
当Servlet容器启动或终止Web应用时,会触发ServletContextEvent事件,该事件由ServletContextListener来处理。在ServletContextListener接口中定义了处理ServletContextEvent事件的两个方法。
- /**
- *当Servlet容器启动Web应用时调用该方法。在调用完该方法之后,容器再对Filter初始化,
- *并且对那些在Web应用启动时就需要被初始化的Servlet进行初始化。
- */
- contextInitialized(ServletContextEventsce)
- /**
- *当Servlet容器终止Web应用时调用该方法。在调用该方法之前,容器会先销毁所有的Servlet和Filter过滤器。
- */
- contextDestroyed(ServletContextEventsce)
下面通过两个具体的例子来介绍ServletContextListener的用法。
例一:在服务启动时,将数据库中的数据加载进内存,并将其赋值给一个属性名,其它的Servlet就可以通过getAttribute进行属性值的访问。
有如下两个步骤:
1:ServletContext对象是一个为整个web应用提供共享的内存,任何请求都可以访问里面的内容
2:如何实现在服务启动的时候就动态的加入到里面的内容:我们需要做的有:
1)实现servletContextListerner接口并将要共享的通过setAttribute(name,data)方法提交到内存中去;
2)应用项目通过getAttribute(name)将数据取到。
- publicclassServletContextLTestimplementsServletContextListener{
- //实现其中的销毁函数
- publicvoidcontextDestroyed(ServletContextEventsce){
- System.out.println("thisislastdestroyeed");
- }
- //实现其中的初始化函数,当有事件发生时即触发
- publicvoidcontextInitialized(ServletContextEventsce){
- ServletContextsct=sce.getServletContext();
- Map<Integer,String>depts=newHashMap<Integer,String>();
- Connectionconnection=null;
- PreparedStatementpstm=null;
- ResultSetrs=null;
- try{
- connection=ConnectTool.getConnection();
- Stringsql="selectdeptNo,dnamefromdept";
- pstm=connection.prepareStatement(sql);
- rs=pstm.executeQuery();
- while(rs.next()){
- depts.put(rs.getInt(1),rs.getString(2));
- }
- //将所取到的值存放到一个属性键值对中
- sct.setAttribute("dept",depts);
- System.out.println("======listenertestisbeginning=========");
- }catch(Exceptione){
- e.printStackTrace();
- }finally{
- ConnectTool.releasersc(rs,pstm,connection);
- }
- }
- }
在完成上述编码后,仍需在web.xml中进行如下配置,以使得该监听器可以起作用。
- <listener>
- <listener-class>ServletContextTest.ServletContextLTest</listener-class>
- </listener>
在完成上述配置后,web服务器在启动时,会直接加载该监听器,通过以下的应用程序就可以进行数据的访问。
- publicclassCreateEmployeeextendsHttpServlet{
- @Override
- protectedvoidservice(HttpServletRequestrequest,HttpServletResponseresponse)
- throwsServletException,IOException{
- ServletContextsct=getServletConfig().getServletContext();
- //从上下文环境中通过属性名获取属性值
- Map<Integer,String>dept=(Map<Integer,String>)sct.getAttribute("dept");
- Set<Integer>key=dept.keySet();
- response.setContentType("text/html;charset=utf-8");
- PrintWriterout=response.getWriter();
- out.println("<html>");
- out.println("<body>");
- out.println("<formaction='/register'action='post'>");
- out.println("<tablealignb='center'>");
- out.println("<tr>");
- out.println("<td>");
- out.println("username:");
- out.println("</td>");
- out.println("<td>");
- out.println("<inputtype='text'name='username'");
- out.println("</tr>");
- out.println("<tr>");
- out.println("<td>");
- out.println("city:");
- out.println("</td>");
- out.println("<td>");
- out.println("<selectname='dept'");
- for(Integeri:key){
- out.println("<optionvalue='"+i+"'>"+dept.get(i)+"</option>");
- }
- out.println("</select>");
- out.println("</td>");
- out.println("<tr>");
- out.println("</table>");
- out.println("</form>");
- out.println("</body>");
- out.println("</html>");
- out.flush();
- }
- }
例二:书写一个类用于统计当Web应用启动后,网页被客户端访问的次数。如果重新启动Web应用,计数器不会重新从1开始统计访问次数,而是从上次统计的结果上进行累加。
在实际应用中,往往需要统计自Web应用被发布后网页被客户端访问的次数,这就要求当Web应用被终止时,计数器的数值被永久存储在一个文件中或者数据库中,等到Web应用重新启动时,先从文件或数据库中读取计数器的初始值,然后在此基础上继续计数。
向文件中写入或读取计数器的数值的功能可以由自定义的MyServletContextListener类来完成,它具有以下功能:
1、在Web应用启动时从文件中读取计数器的数值,并把表示计数器的Counter对象存放到Web应用范围内。存放计数器的文件的路径为helloapp/count/count.txt。
2、在Web应用终止时把Web应用范围内的计数器的数值保存到count.txt文件中。
- publicclassMyServletContextListenerimplementsServletContextListener{
- publicvoidcontextInitialized(ServletContextEventsce){
- System.out.println("helloappapplicationisInitialized.");
- //获取ServletContext对象
- ServletContextcontext=sce.getServletContext();
- try{
- //从文件中读取计数器的数值
- BufferedReaderreader=newBufferedReader(
- newInputStreamReader(context.
- getResourceAsStream("/count/count.txt")));
- intcount=Integer.parseInt(reader.readLine());
- reader.close();
- //创建计数器对象
- Countercounter=newCounter(count);
- //把计数器对象保存到Web应用范围
- context.setAttribute("counter",counter);
- }catch(IOExceptione){
- e.printStackTrace();
- }
- }
- publicvoidcontextDestroyed(ServletContextEventsce){
- System.out.println("helloappapplicationisDestroyed.");
- //获取ServletContext对象
- ServletContextcontext=sce.getServletContext();
- //从Web应用范围获得计数器对象
- Countercounter=(Counter)context.getAttribute("counter");
- if(counter!=null){
- try{
- //把计数器的数值写到count.txt文件中
- Stringfilepath=context.getRealPath("/count");
- filepath=filepath+"/count.txt";
- PrintWriterpw=newPrintWriter(filepath);
- pw.println(counter.getCount());
- pw.close();
- }catch(IOExceptione){
- e.printStackTrace();
- }
- }
- }
- }
将用户自定义的MyServletContextListener监听器在Servlet容器进行注册,Servlet容器会在启动或终止Web应用时,会调用该监听器的相关方法。在web.xml文件中,<listener>元素用于向容器注册监听器:
- <listener>
- <listener-class>ServletContextTest.MyServletContextListener<listener-class/>
- </listener>
通过上述两个例子,即可以非常清楚的了解到ServletContextListener接口的使用方法及技巧。
在Container加载Web应用程序时(例如启动Container之后),会呼叫contextInitialized(),而当容器移除Web应用程序时,会呼叫contextDestroyed ()方法。
通过Tomcat控制台的打印结果的先后顺序,会发现当Web应用启动时,Servlet容器先调用contextInitialized()方法,再调用lifeInit的init()方法;
当Web应用终止时,Servlet容器先调用lifeInit的destroy()方法,再调用contextDestroyed()方法。
由此可见,在Web应用的生命周期中,ServletContext对象最早被创建,最晚被销毁。
原文地址:http://blog.youkuaiyun.com/zhaozheng7758/archive/2010/12/28/6103700.aspx