1. 程式人生 > >Tomcat啟動時自動載入Servlet

Tomcat啟動時自動載入Servlet

想實現這樣的功能:

      1.   Tomcat啟動時隨即啟動Servlet;

      2.   Servlet啟動時定時執行一個任務。

要點:

      1。Tomcat中啟動Servlet時,只需要在Servlet所在的工程的配置檔案web.xml中寫成如下即可;

     <servlet>

          <servlet-name>TimeServlet</servlet-name>
          <servlet-class>TimeServlet</servlet-class>
          <load-on-startup>1</load-on-startup>
     </servlet>

     <servlet-mapping>
          <servlet-name>TimeServlet</servlet-name>
          <url-pattern>/servlet/TimeServlet</url-pattern>
     </servlet-mapping>

       所以最主要的是加一句:<load-on-startup>1</load-on-startup>

       2。在Servlet中寫一個定時器和要執行的任務即可。

       我的Servlet叫這個名字: TimeServlet,定時輸出一句話,System.out.println("____ok____");也就是用來測試一下定時任務執行情況正常而已。

Java程式碼  收藏程式碼
  1. import java.io.IOException;  
  2. import java.util.Timer;  
  3. import javax.servlet.ServletException;  
  4. import javax.servlet.http.HttpServlet;  
  5. import javax.servlet.http.HttpServletRequest;  
  6. import javax.servlet.http.HttpServletResponse;  
  7. publicclass TimeServlet extends HttpServlet   
  8. {     
  9.     privatestaticfinallong serialVersionUID = 1L;  
  10.     public TimeServlet()   
  11.     {  
  12.         super();  
  13.     }  
  14.     publicvoid init() throws ServletException   
  15.     {  
  16.         Timer timer = new Timer();  
  17.         timer.schedule(new MyTask(), 10002000);// 在1秒後執行此任務,每次間隔2秒,如果傳遞一個Data引數,就可以在某個固定的時間執行這個任務.
  18.         // 這個是用來停止此任務的,否則就一直迴圈執行此任務了
  19.         while (true)   
  20.         {  
  21.             try
  22.             {  
  23.                 int ch = System.in.read(); //輸入字元“c”,回車即停止執行
  24.                 if (ch - 'c' == 0)   
  25.                 {  
  26.                     timer.cancel();// 使用這個方法退出任務
  27.                 }  
  28.             } catch (IOException e)   
  29.             {  
  30.                 e.printStackTrace();  
  31.             }  
  32.         }  
  33.     }  
  34.     //具體執行的任務
  35.     class MyTask extends java.util.TimerTask   
  36.     {  
  37.         publicvoid run()   
  38.         {  
  39.             System.out.println("____ok____");  
  40.         }  
  41.     }  
  42.     publicvoid destroy()   
  43.     {  
  44.         super.destroy();   
  45.     }  
  46.     publicvoid doGet(HttpServletRequest request, HttpServletResponse response)  
  47.             throws ServletException, IOException   
  48.             {  
  49.     }  
  50.     publicvoid doPost(HttpServletRequest request, HttpServletResponse response)  
  51.             throws ServletException, IOException   
  52.             {  
  53.     }  
  54. }  

有關定時器的知識補充:

Java程式碼  收藏程式碼
  1. import java.io.IOException;  
  2. import java.util.Timer;  
  3. publicclass TimerTest   
  4. {  
  5.     publicstaticvoid main(String[] args)   
  6.     {  
  7.         Timer timer = new Timer();  
  8.         timer.schedule(new MyTask(), 10002000);// 在1秒後執行此任務,每次間隔2秒,如果傳遞一個Data引數,就可以在某個固定的時間執行這個任務.
  9.         // 這個是用來停止此任務的,否則就一直迴圈執行此任務了
  10.         while (true)   
  11.         {  
  12.             try
  13.             {  
  14.                 int ch = System.in.read();  
  15.                 if (ch - 'c' == 0)   
  16.                 {  
  17.                     timer.cancel();// 使用這個方法退出任務
  18.                 }  
  19.             } catch (IOException e)   
  20.             {  
  21.                 e.printStackTrace();  
  22.             }  
  23.         }  
  24.     }  
  25.     staticclass MyTask extends java.util.TimerTask   
  26.     {  
  27.         publicvoid run()   
  28.         {  
  29.             System.out.println("____ok____");  
  30.         }  
  31.     }  
  32. }  

java定時器的使用【轉載】

java定時器的使用

定時器類Timer在java.util包中。使用時,先例項化,然後使用例項的schedule(TimerTask task, long delay)方法,設定指定的任務task在指定的延遲delay後執行。定時器任務類TimerTask是抽象類,繼承並重寫其run()方法,可實現具體任務。

schedule(TimerTask task, Date time)設定指定任務task在指定時間time執行。

cancel()方法結束這個定時器。

schedule(TimerTask task, long delay, long period)方法設定指定任務task在指定延遲delay後進行固定延遲peroid的執行。

scheduleAtFixedRate(TimerTask task, long delay, long period)方法設定指定任務task在指定延遲delay後進行固定頻率peroid的執行。

要實現一個定時任務,運用java中的Timer和TimerTask類可以非常容易實現實時呼叫處理函式。這兩個類使用起來非常方便,可以完成我們對定時器的絕大多數需要。

看個簡單的例子:

imp<wbr>ort java.io.IOException;</wbr>

imp<wbr>ort java.util.Timer;</wbr>

public class TimerTest {

public static void main(String[] args){

Timer timer = new Timer();

timer.schedule(new MyTask(), 1000, 2000);//在1秒後執行此任務,每次間隔2秒,如果傳遞一個Da<wbr>ta引數,就可以在某個固定的時間執行這個任務.</wbr>

while(true){//這個是用來停止此任務的,否則就一直迴圈執行此任務了

try {

int ch = System.in.read();

if(ch-'c'==0){

timer.cancel();//使用這個方法退出任務

}

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

static class MyTask extends java.util.TimerTask{

@Override

public void run() {

// TODO Auto-generated method stub

System.out.println("________");

}

}

}

Servlet偵聽器結合Java定時器實現任務計劃排程[轉]

好多朋友用過Windows的任務計劃,也有不少程式迷自己曾寫過時鐘報警、系統自動關機等趣味程式,可卻很少有朋友在Web工程中實現過類似功能。今天有空把筆者先前曾在Tomcat上實現的類似功能,搬出來與大家共享。

    早在幾年前,我公司跟某市財政局合作專案開發,為加強財政局對所屬單位財務狀況的有效監管,開發、實施了財政局資料中心專案。此專案採用B/S加C/S混合結構模式。財政局Web伺服器上架設資料同步接收裝置,由市屬單位每天下班前把財務資訊通過HTTP協議上傳至財政局中心伺服器,與Web伺服器上的接收裝置對接。財政局內部各部門需要查閱大量財務資訊,獲取完備的市屬單位當前財務狀況資訊,各部門按職能劃分,需要準確的獲取各部門各自所關注的彙總資訊,以財政報表的形式提供。 

    因財政資料量大,實時計算財政報表速度較慢,當初就考慮用報表快取來減輕伺服器的負擔,但用快取需要一個合理的快取更新機制。考慮到各市屬單位每天下班前才把財務資料上傳,財政局每天所檢視到的財務資訊其實並不包括當天(除非有某位領導等到所屬單位全部上傳完之後才來檢視資訊,應該已經下班了),所以要是能實現任務計劃排程,在每晚深夜把當天及歷史財務資訊彙總,更新快取,速度瓶頸不就解決了嗎。

    當時由於系統核心是基於Web部署的,報表計算引擎也相應的部署在Tomcat容器上,因此如果想要借用Windows的任務計劃來實現定時計算,就需要額外編寫普通桌面應用程式介面,稍顯複雜。於是就琢磨著想在Web上實現,經過查閱較多相關資料,發現Java定時器(java.util.Timer)有定時觸發計劃任務的功能,通過配置定時器的間隔時間,在某一間隔時間段之後會自動有規律的呼叫預先所安排的計劃任務(java.util.TimerTask)。另外,由於我們希望當Web工程啟動時,定時器能自動開始計時,在整個Web工程的生命期裡,定時器能在每晚深夜觸發一次報表計算引擎。因此定時器的存放位置也值得考查,不能簡單的存在於單個Servlet或JavaBean中,必須能讓定時器宿主的存活期為整個Web工程生命期,在工程啟動時能自動載入執行。結合這兩點,跟Servlet上下文有關的偵聽器就最合適不過了,通過在工程的配置檔案中加以合理配置,會在工程啟動時自動執行,並在整個工程生命期中處於監聽狀態。

    下面就Servlet偵聽器結合Java定時器來講述整個實現過程。要運用Servlet偵聽器需要實現javax.servlet.ServletContextListener介面,同時實現它的contextInitialized(ServletContextEvent event)和contextDestroyed(ServletContextEvent event)兩個介面函式。考慮定時器有個建立和銷燬的過程,看了前面兩個介面函式,就不容置疑的把建立的過程置入contextInitialized,把銷燬的過程置入contextDestroyed了。

    我把ServletContextListener的實現類取名為ContextListener,在其內新增一個定時器,示例程式碼如下所示(為考慮篇幅,僅提供部分程式碼供讀者參考):

  1.     private java.util.Timer timer = null;
  2.     public void contextInitialized(ServletContextEvent event) {
  3.         timer = new java.util.Timer(true);
  4.         event.getServletContext().log("定時器已啟動");        
  5.          timer.schedule(new MyTask(event.getServletContext()), 0, 60*60*1000);
  6.         event.getServletContext().log("已經新增任務排程表");
  7.     }
  8.     public void contextDestroyed(ServletContextEvent event) {
  9.         timer.cancel();
  10.         event.getServletContext().log("定時器銷燬");
  11.     }

    以上程式碼中, timer.schedule(new MyTask(event.getServletContext()), 0, 60*60*1000)這一行為定時器排程語句,其中MyTask是自定義需要被排程的執行任務(在我的財政資料中心專案中就是報表計算引擎入口),從java.util.TimerTask繼承,下面會重點講述,第三個引數表示每小時(即60*60*1000毫秒)被觸發一次,中間引數0表示無延遲。其它程式碼相當簡單,不再詳細說明。

   下面介紹MyTask的實現,上面的程式碼中看到了在構造MyTask時,傳入了javax.servlet.ServletContext型別引數,是為記錄Servlet日誌方便而傳入,因此需要過載MyTask的建構函式(其父類java.util.TimerTask原建構函式是沒有引數的)。在timer.schedule()的排程中,設定了每小時排程一次,因此如果想實現排程任務每24小時被執行一次,還需要判斷一下時鐘點,以常量C_SCHEDULE_HOUR表示(晚上12點,也即0點)。同時為防止24小時執行下來,任務還未執行完(當然,一般任務是沒有這麼長的),避免第二次又被排程以引起執行衝突,設定了當前是否正在執行的狀態標誌isRunning。示例程式碼如下所示:

  1.     private static final int C_SCHEDULE_HOUR   = 0;
  2.     private static boolean isRunning = false;
  3.          private ServletContext context = null;
  4.     public MyTask(ServletContext context) {
  5.         this.context = context;
  6.     }
  7.     public void run() {
  8.         Calendar cal = Calendar.getInstance();        
  9.         if (!isRunning)  {           
  10.             if (C_SCHEDULE_HOUR == cal.get(Calendar.HOUR_OF_DAY)) {            
  11.                     isRunning = true;                
  12.                 context.log("開始執行指定任務");
  13.                 //TODO 新增自定義的詳細任務,以下只是示例
  14.                 int i = 0;
  15.                 while (i++ < 10) {
  16.                     context.log("已完成任務的" + i + "/" + 10);
  17.                 }
  18.                 isRunning = false;
  19.                 context.log("指定任務執行結束");               
  20.             }            
  21.         } else {
  22.             context.log("上一次任務執行還未結束");
  23.         }
  24.     }

    上面程式碼中“//TODO……”之下四行是真正被排程執行的演示程式碼(在我的財政資料中心專案中就是報表計算過程),您可以換成自己希望執行的語句。

到這兒,ServletContextListener和MyTask的程式碼都已完整了。最後一步就是把ServletContextListener部署到您的Web工程中去,在您工程的web.xml配置檔案中加入如下三行:

     <listener></listener>

         <listener>com.test.ContextListener</listener>

    當然,上面的com.test得換成您自己的包名了。儲存web.xml檔案後,把工程打包部署到Tomcat中即可。任務會在每晚12點至凌晨1點之間被執行,上面的程式碼會在Tomcat的日誌檔案中記錄如下:

2003-12-05 0:21:39 開始執行指定任務

2003-12-05 0:21:39 已完成任務的1/10

    ……

2003-12-05 0:21:39 已完成任務的10/10

2003-12-05 0:21:39 指定任務執行結束