實驗中發現在 ASP.NET 中可以使用計時器(Timer)完成一些定時動作。這一點可能會對我們的一些 Web 程式有益。
下面首先介紹測試使用的一個例子:
- 首先在 global.asax 中的 Application_OnStart 事件程序中定義計時器,代碼如下:
[C#] global.asax
然後我們簡單寫一個 test.aspx 來查看 Application("TimeStamp") 的值。代碼如下:
分析:
根 據 global.asax 中的代碼,我們設定了一個計時器,每隔 10 秒鐘執行一次 Fresher() 過程;在 Fresher() 過程中我們事實上只是重新寫入了一個 Application("TimeStamp") 新值。換句話說,Application("TimeStamp") 的值是應該每隔 10 秒鐘更新一次的。
是不是這樣的呢?通過 test.aspx 的反覆重新整理觀察 Application("TimeStamp") 的值,的確發現這個值在每隔 10 秒地變化一次,而其他時候則保持不變。與我們的預期是一致的。
意義:
通 過引入計時器我們可以在 ASP.NET 的全域性程式(Application)中靈活的使用計時器完成一些定時操作,比如:在社區/論壇系統中,每隔 5 分鐘更新一次線上使用者列表,每隔 1 個小時更新一次使用者經驗值,或者每隔一天備份一次關鍵資料等等。這個思路應該是很誘人的。
探討:
Q: 是否在 ASP.NET 代碼的任何地方都可以使用計時器呢?
A: 我沒有測試過在普通 *.aspx 中插入計時器的情形。但從 B/S 程式的特點來看,即使在 *.aspx 中插入計時器可行,也不是一種好的選擇。因為對於 B/S 程式來說,伺服器接到用戶端的請求本身就是一個事件,在這個事件處理過程中,伺服器必須迅速的作出回應,為用戶端產生相應的 HTML 程式碼,然後結束這一過程。如果在 *.aspx 使用計時器(如果允許的話),則第一沒有太大必要,第二很容易使系統因為插入的計時器過多(因為每一次 *.aspx 的執行都有可能插入一個新的計時器)而使系統癱瘓。
因此,我建議只在 global.asax 的 Application_OnStart 中使用比較安全一些。歡迎對此感興趣的朋友對此發表見解。
2.提取了asp.net forums中定時器的應用,並分析了在Asp.Net Forums中,對定時器有如下應用:
1. 更新論壇統計資訊
2. 定時索引指定條數的文章
3. 定時群發隊列中的郵件
Forums中對定時器的調用是放在自訂HttpModule的Init方法中(如果您沒有使用HttpModule,也可以在Globals.aspx中的Application_OnStart 中調用定時器)。
// 定時器
static Timer statsTimer;
static Timer emailTimer;
// 定時間隔
private long EmailInterval = ForumConfiguration.GetConfig().ThreadIntervalEmail * 60000;
private long StatsInterval = ForumConfiguration.GetConfig().ThreadIntervalStats * 60000;
public String ModuleName {
get { return "ForumsHttpModule"; }
}
// *********************************************************************
// ForumsHttpModule
//
/// <summary>
/// Initializes the HttpModule and performs the wireup of all application
/// events.
/// </summary>
/// <param name="application">Application the module is being run for</param>
public void Init(HttpApplication application) {
// Wire-up application events
//
// 略去其他代碼
ForumConfiguration forumConfig = ForumConfiguration.GetConfig();
// 如果使用定時器並且定時器還沒初始化
if( forumConfig != null
&& forumConfig.IsBackgroundThreadingDisabled == false ) {
if (emailTimer == null)
// 建立定時器
// 建立一個TimerCallback委託,具體要執行的方法在ScheduledWorkCallbackEmailInterval中
emailTimer = new Timer(new TimerCallback(ScheduledWorkCallbackEmailInterval), application.Context, EmailInterval, EmailInterval);
if( forumConfig.IsIndexingDisabled == false
&& statsTimer == null ) {
statsTimer = new Timer(new TimerCallback(ScheduledWorkCallbackStatsInterval), application.Context, StatsInterval, StatsInterval);
}
}
}
/// <summary>
/// 釋放定時器
/// </summary>
public void Dispose() {
statsTimer = null;
emailTimer = null;
}
#region Timer Callbacks
/// <summary>
/// 定時發送隊列中待發送的郵件
/// </summary>
private void ScheduledWorkCallbackEmailInterval (object sender) {
try {
// 當處理郵件時暫停定時器
emailTimer.Change( System.Threading.Timeout.Infinite, EmailInterval );
// 發送隊列中的郵件
//
Emails.SendQueuedEmails( (HttpContext) sender);
// 更新匿名使用者
//
Users.UpdateAnonymousUsers( (HttpContext) sender);
}
catch( Exception e ) {
ForumException fe = new ForumException( ForumExceptionType.EmailUnableToSend, "Scheduled Worker Thread failed.", e );
fe.Log();
}
finally {
// 重新啟動定時器
emailTimer.Change( EmailInterval, EmailInterval );
}
}
/// <summary>
/// 定時索引文章和定時更新論壇統計資訊
/// </summary>
private void ScheduledWorkCallbackStatsInterval(object sender) {
try {
// 休眠定時器
statsTimer.Change( System.Threading.Timeout.Infinite, StatsInterval );
// 每次索引100篇文章
//
Search.IndexPosts( (HttpContext) sender, 100);
// 更新論壇統計資訊
SiteStatistics.LoadSiteStatistics( (HttpContext) sender, true, 1 );
}
catch( Exception e ) {
ForumException fe = new ForumException( ForumExceptionType.UnknownError, "Failure performing scheduled statistics maintenance.", e );
fe.Log();
}
finally {
// 喚醒定時器
statsTimer.Change( StatsInterval, StatsInterval);
}
}
#endregion