根據不同需求,在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>
