在Servlet API中有一個(gè)ServletContextListener接口,它能夠監(jiān)聽ServletContext對象的生命周期,實(shí)際上就是監(jiān)聽Web應(yīng)用的生命周期。
當(dāng)Servlet容器啟動(dòng)或終止Web應(yīng)用時(shí),會(huì)觸發(fā)ServletContextEvent事件,該事件由ServletContextListener 來處理。在 ServletContextListener 接口中定義了處理ServletContextEvent事件的兩個(gè)方法。
l contextInitialized(ServletContextEventsce):當(dāng)Servlet容器啟動(dòng)Web應(yīng)用時(shí)調(diào)用該方法。在調(diào)用完該方法之后,容器再對Filter初始化,并且對那些在Web應(yīng)用啟動(dòng)時(shí)就需要被初始化的Servlet進(jìn)行初始化。
l contextDestroyed(ServletContextEventsce):當(dāng)Servlet容器終止Web應(yīng)用時(shí)調(diào)用該方法。在調(diào)用該方法之前,容器會(huì)先銷毀所有的Servlet和Filter過濾器。
下面通過兩個(gè)具體的例子來介紹ServletContextListener的用法。
例一:在服務(wù)啟動(dòng)時(shí),將數(shù)據(jù)庫中的數(shù)據(jù)加載進(jìn)內(nèi)存,并將其賦值給一個(gè)屬性名,其它的Servlet就可以通過getAttribute進(jìn)行屬性值的訪問。有如下兩個(gè)步驟:
1:ServletContext對象是一個(gè)為整個(gè)web應(yīng)用提供共享的內(nèi)存,任何請求都可以訪問里面的內(nèi)容
2:如何實(shí)現(xiàn)在服務(wù)啟動(dòng)的時(shí)候就動(dòng)態(tài)的加入到里面的內(nèi)容:我們需要做的有:
1)實(shí)現(xiàn)servletContextListerner接口 并將要共享的通過setAttribute(name,data)方法提交到內(nèi)存中去 ;
2)應(yīng)用項(xiàng)目通過getAttribute(name)將數(shù)據(jù)取到。
packageServletContextTest;
importjava.sql.Connection;
importjava.sql.PreparedStatement;
importjava.sql.ResultSet;
importjava.util.HashMap;
importjava.util.Map;
importjavax.servlet.ServletContext;
importjavax.servlet.ServletContextEvent;
importjavax.servlet.ServletContextListener;
importutil.ConnectTool;
publicclass ServletContextLTest implements ServletContextListener{
//實(shí)現(xiàn)其中的銷毀函數(shù)
public void contextDestroyed(ServletContextEventsce) {
System.out.println("thisis last destroyeed");
}
//實(shí)現(xiàn)其中的初始化函數(shù),當(dāng)有事件發(fā)生時(shí)即觸發(fā)
public voidcontextInitialized(ServletContextEvent sce) {
ServletContextsct=sce.getServletContext();
Map<Integer,String> depts=newHashMap<Integer,String>();
Connection connection=null;
PreparedStatement pstm=null;
ResultSet rs=null;
try{
connection=ConnectTool.getConnection();
String sql="select deptNo,dnamefrom dept";
pstm=connection.prepareStatement(sql);
rs=pstm.executeQuery();
while(rs.next()){
depts.put(rs.getInt(1),rs.getString(2));
}
//將所取到的值存放到一個(gè)屬性鍵值對中
sct.setAttribute("dept",depts);
System.out.println("======listener test is beginning=========");
}catch(Exception e){
e.printStackTrace();
}finally{
ConnectTool.releasersc(rs, pstm,connection);
}
}
}
在完成上述編碼后,仍需在web.xml中進(jìn)行如下配置,以使得該監(jiān)聽器可以起作用。
<listener>
<listener-class>ServletContextTest.ServletContextLTest</listener-class>
</listener>
在完成上述配置后,web服務(wù)器在啟動(dòng)時(shí),會(huì)直接加載該監(jiān)聽器,通過以下的應(yīng)用程序就可以進(jìn)行數(shù)據(jù)的訪問。
packageServletContextTest;
importjava.io.IOException;
importjava.io.PrintWriter;
importjava.util.*;
importjavax.servlet.ServletContext;
importjavax.servlet.ServletException;
importjavax.servlet.http.HttpServlet;
importjavax.servlet.http.HttpServletRequest;
importjavax.servlet.http.HttpServletResponse;
publicclass CreateEmployee extends HttpServlet{
@Override
protected void service(HttpServletRequestrequest, HttpServletResponse response)
throws ServletException,IOException {
ServletContextsct=getServletConfig().getServletContext();
//從上下文環(huán)境中通過屬性名獲取屬性值
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("<input type='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(Integer i: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();
}
}
例二:書寫一個(gè)類用于統(tǒng)計(jì)當(dāng)Web應(yīng)用啟動(dòng)后,網(wǎng)頁被客戶端訪問的次數(shù)。如果重新啟動(dòng)Web應(yīng)用,計(jì)數(shù)器不會(huì)重新從1開始統(tǒng)計(jì)訪問次數(shù),而是從上次統(tǒng)計(jì)的結(jié)果上進(jìn)行累加。在實(shí)際應(yīng)用中,往往需要統(tǒng)計(jì)自Web應(yīng)用被發(fā)布后網(wǎng)頁被客戶端訪問的次數(shù),這就要求當(dāng)Web應(yīng)用被終止時(shí),計(jì)數(shù)器的數(shù)值被永久存儲(chǔ)在一個(gè)文件中或者數(shù)據(jù)庫中,等到Web應(yīng)用重新啟動(dòng)時(shí),先從文件或數(shù)據(jù)庫中讀取計(jì)數(shù)器的初始值,然后在此基礎(chǔ)上繼續(xù)計(jì)數(shù)。
向文件中寫入或讀取計(jì)數(shù)器的數(shù)值的功能可以由自定義的 MyServletContextListener類來完成,它具有以下功能:
1、在 Web 應(yīng)用啟動(dòng)時(shí)從文件中讀取計(jì)數(shù)器的數(shù)值,并把表示計(jì)數(shù)器的 Counter對象存放到 Web 應(yīng)用范圍內(nèi)。存放計(jì)數(shù)器的文件的路徑為helloapp/count/count.txt。
2、在Web應(yīng)用終止時(shí)把Web應(yīng)用范圍內(nèi)的計(jì)數(shù)器的數(shù)值保存到count.txt文件中。
packageServletContextTest;
importjavax.servlet.ServletContext;
importjavax.servlet.ServletContextEvent;
importjavax.servlet.ServletContextListener;
publicclass MyServletContextListener implements ServletContextListener{
public voidcontextInitialized(ServletContextEvent sce){
System.out.println("helloappapplication is Initialized.");
//獲取ServletContext對象
ServletContextcontext=sce.getServletContext();
try{
//從文件中讀取計(jì)數(shù)器的數(shù)值
BufferedReader reader=newBufferedReader(
new InputStreamReader(context.
getResourceAsStream("/count/count.txt")));
intcount=Integer.parseInt(reader.readLine());
reader.close();
//創(chuàng)建計(jì)數(shù)器對象
Counter counter=new Counter(count);
//把計(jì)數(shù)器對象保存到Web應(yīng)用范圍
context.setAttribute("counter",counter);
} catch(IOException e) {
e.printStackTrace();
}
}
public voidcontextDestroyed(ServletContextEvent sce){
System.out.println("helloappapplication is Destroyed.");
//獲取ServletContext對象
ServletContextcontext=sce.getServletContext();
//從Web應(yīng)用范圍獲得計(jì)數(shù)器對象
Countercounter=(Counter)context.getAttribute("counter");
if(counter!=null){
try{
//把計(jì)數(shù)器的數(shù)值寫到count.txt文件中
Stringfilepath=context.getRealPath("/count");
filepath=filepath+"/count.txt";
PrintWriter pw=newPrintWriter(filepath);
pw.println(counter.getCount());
pw.close();
} catch(IOException e) {
e.printStackTrace();
}
}
}
}
將用戶自定義的MyServletContextListener監(jiān)聽器在Servlet容器進(jìn)行注冊,Servlet容器會(huì)在啟動(dòng)或終止Web應(yīng)用時(shí),會(huì)調(diào)用該監(jiān)聽器的相關(guān)方法。在web.xml文件中,<listener>元素用于向容器注冊監(jiān)聽器:
<listener>
<listener-class>ServletContextTest.MyServletContextListener<listener-class/>
</listener>
通過上述兩個(gè)例子,即可以非常清楚的了解到ServletContextListener接口的使用方法及技巧。在Container加載Web應(yīng)用程序時(shí)(例如啟動(dòng) Container之后),會(huì)呼叫contextInitialized(),而當(dāng)容器移除Web應(yīng)用程序時(shí),會(huì)呼叫contextDestroyed ()方法。通過Tomcat 控制臺的打印結(jié)果的先后順序,會(huì)發(fā)現(xiàn)當(dāng) Web 應(yīng)用啟動(dòng)時(shí),Servlet容器先調(diào)用contextInitialized()方法,再調(diào)用lifeInit的init()方法;當(dāng)Web應(yīng)用終止時(shí),Servlet容器先調(diào)用lifeInit的destroy()方法,再調(diào)用contextDestroyed()方法。由此可見,在Web應(yīng)用的生命周期中,ServletContext對象最早被創(chuàng)建,最晚被銷毀。
聯(lián)系客服