方法一、 using System; using System.Data; using System.Configuration; using System.Collections; using System.Web; using System.Web.Security; using System.Web.SessionState; using System.Timers; using System.Net; using System.IO; using System.Text; using System.Threading; namespace qumiao.com { public class Global : System.Web.HttpApplication { protected void Application_Start(object sender, EventArgs e) { //定義定時器 System.Timers.Timer myTimer = new System.Timers.Timer(5000); myTimer.Elapsed += new ElapsedEventHandler(myTimer_Elapsed); myTimer.Enabled = true; myTimer.AutoReset = true; } void myTimer_Elapsed(object source, ElapsedEventArgs e) { try { Log.SaveNote(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + ":AutoTask is Working!"); YourTask(); } catch (Exception ee) { Log.SaveException(ee); } } void YourTask() { //在這裡寫你需要執行的任務 } protected void Application_End(object sender, EventArgs e) { Log.SaveNote(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + ":Application End!"); //下面的代碼是關鍵,可解決IIS應用程式集區自動回收的問題 Thread.Sleep(1000); //這裡設定你的web地址,可以隨便指向你的任意一個aspx頁面甚至不存在的頁面,目的是要激發Application_Start string url = http://www.shaoqun.com HttpWebRequest myHttpWebRequest = (HttpWebRequest)WebRequest.Create(url); HttpWebResponse myHttpWebResponse = (HttpWebResponse)myHttpWebRequest.GetResponse(); Stream receiveStream = myHttpWebResponse.GetResponseStream();//得到回寫的位元組流 } } } 原理:Global.asax 可以是asp.net中應用程式或會話事件處理常式,我們用到了Application_Start(應用程式開始事件)和Application_End(應用程式結束事件)。當應用程式開始時,啟動一個定時器,用來定時執行任務YourTask()方法,這個方法裡面可以寫上需要調用的邏輯代碼,可以是單線程和多線程。當應用程式結束時,如IIS的應用程式集區回收,讓asp.net去訪問當前的這個web地址。這裡需要訪問一個aspx頁面,這樣就可以重新啟用應用程式。Log類是一個記錄日誌的一個類,下面是測試時產生的日誌資訊: ================================================================ 2008-10-30 17:46:10:AutoTask is Working! 2008-10-30 17:46:15:AutoTask is Working! 2008-10-30 17:46:20:AutoTask is Working! 2008-10-30 17:46:23:Application End! 2008-10-30 17:46:29:AutoTask is Working! 2008-10-30 17:46:34:AutoTask is Working! 從日誌中發現,當手動回收IIS的應用程式集區之後,計劃任務還在執行,說明我們的目的達到了。 如果將Application_End中的代碼注釋掉,會發現Application End之後,計劃任務停止工作了,如下: ================================================================ 2008-10-30 18:01:34:AutoTask is Working! 2008-10-30 18:01:39:AutoTask is Working! 2008-10-30 18:01:44:AutoTask is Working! 2008-10-30 18:01:46:Application End! 局限性:可以解決應用程式集區自動或者手動回收,但是無法解決IIS重啟或者web伺服器重啟的問題,當然這種情況出現的時候不多,而且如果有人訪問你的網站的時候,又會自動啟用計劃任務了。 方案二、 <%@ Application Language="C#" %> <%@ import Namespace="System.IO" %> <script runat="server"> void Application_Start(object sender, EventArgs e) { // 在應用程式啟動時啟動並執行代碼 System.Timers.Timer myTimer = new System.Timers.Timer(10000); myTimer.Elapsed += new System.Timers.ElapsedEventHandler(OnTimedEvent); myTimer.Interval = 10000; myTimer.Enabled = true; } void Application_End(object sender, EventArgs e) { // 在應用程式關閉時啟動並執行代碼 } void Application_Error(object sender, EventArgs e) { // 在出現未處理的錯誤時啟動並執行代碼 } void Session_Start(object sender, EventArgs e) { // 在新會話啟動時啟動並執行代碼 } void Session_End(object sender, EventArgs e) { // 在會話結束時啟動並執行代碼。 // 注意: 只有在 Web.config 檔案中的 sessionstate 模式設定為 // InProc 時,才會引發 Session_End 事件。如果會話模式設定為 StateServer // 或 SQLServer,則不會引發該事件。 } private static void OnTimedEvent(object source, System.Timers.ElapsedEventArgs e) { //間隔時間執行某動作 //指定記錄檔的目錄 string fileLogPath = AppDomain.CurrentDomain.BaseDirectory + "SystemLog"; string fileLogName = "SoftPrj_CN_" + DateTime.Now.ToLongDateString() + "_log.txt"; //定義檔案資訊對象 FileInfo finfo = new FileInfo(fileLogPath + fileLogName); //建立唯寫檔案流 using (FileStream fs = finfo.OpenWrite()) { //根據上面建立的檔案流建立寫資料流 StreamWriter strwriter = new StreamWriter(fs); //設定寫資料流的起始位置為檔案流的末尾 strwriter.BaseStream.Seek(0, SeekOrigin.End); //寫入錯誤發生時間 strwriter.WriteLine("發生時間: " + DateTime.Now.ToString()); //寫入日誌內容並換行 //strwriter.WriteLine("錯誤內容: " + message); strwriter.WriteLine("錯誤內容: "); //寫入間隔符 strwriter.WriteLine("---------------------------------------------"); strwriter.WriteLine(); //清空緩衝區內容,並把緩衝區內容寫入基礎流 strwriter.Flush(); //關閉寫資料流 strwriter.Close(); fs.Close(); } } </script> 方案三、 <%@ Application Language="C#" %> <%@ Import Namespace="System.IO" %> <%@ Import Namespace="System.Threading" %> <script RunAt="server"> string LogPath; Thread thread; void WriteLog() { while (true) { StreamWriter sw = new StreamWriter(LogPath, true, Encoding.UTF8); sw.WriteLine(thread.Name + ":" + DateTime.Now.ToString()); sw.Close(); Thread.CurrentThread.Join(1000 * 10);//阻止10秒 } } void Application_Start(object sender, EventArgs e) { LogPath = HttpContext.Current.Server.MapPath("log.txt"); //在應用程式啟動時啟動並執行代碼 thread = new Thread(new ThreadStart(WriteLog)); thread.Name = "寫登入日誌線程"; thread.Start(); } void Application_End(object sender, EventArgs e) { // 在應用程式關閉時啟動並執行代碼 } void Application_Error(object sender, EventArgs e) { // 在出現未處理的錯誤時啟動並執行代碼 } void Session_Start(object sender, EventArgs e) { // 在新會話啟動時啟動並執行代碼 } void Session_End(object sender, EventArgs e) { // 在會話結束時啟動並執行代碼。 // 注意: 只有在 Web.config 檔案中的 sessionstate 模式設定為 // InProc 時,才會引發 Session_End 事件。如果會話模式設定為 StateServer // 或 SQLServer,則不會引發該事件。 } </script>