根據不同需求,在web開發中,有時候我們可能要定時的執行一下任務,比如定時發布網站中的文章,這時我們就需要做一個定時執行更新的操作,但由於web的http是無狀態的連接,如何才能時時進行任務的更新呢?方法還是有的,看看下面兩種方法吧!
方法一:
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>