ASP.NET中使用計時器(Timer)來實現群發郵件等功能

iDotNetSpace發表於2009-03-11

實驗中發現在 ASP.NET 中可以使用計時器(Timer)完成一些定時動作。這一點可能會對我們的一些 Web 程式有益。

下面首先介紹測試使用的一個例子:

首先在 global.asax 中的 Application_OnStart 事件過程中定義計時器,程式碼如下:
[C#] global.asax


    void Application_Start(object sender, EventArgs e)
    {
        // 在應用程式啟動時執行的程式碼

        Timer atimer = new Timer(10000);

        atimer.Elapsed += timer_execute;

        atimer.AutoReset = true;
        atimer.Enabled = true;
       
        Application.Lock();
        Application["TimeStamp"] = DateTime.Now.ToString();
        Application.UnLock();
    }

    void timer_execute(object sender, EventArgs e)  //定時執行的函式
    {
        Application.Lock();
        Application["TimeStamp"] = DateTime.Now.ToString();
        Application.UnLock();
    }
              

然後我們簡單寫一個 test.aspx 來檢視 Application("TimeStamp") 的值。程式碼如下:

C#

    Response.Write(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
        //
        ///


        /// Initializes the HttpModule and performs the wireup of all application
        /// events.
        ///

        /// Application the module is being run for
        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);
            }
        }
        }
 
        ///
        /// 釋放定時器
        ///

        public void Dispose() {
            statsTimer = null;
            emailTimer = null;
        }
 
        #region Timer Callbacks
        ///
        /// 定時傳送佇列中待傳送的郵件
        ///

        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 );
            }
        }
 
        ///
        /// 定時索引帖子和定時更新論壇統計資訊
        ///

        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

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/12639172/viewspace-566850/,如需轉載,請註明出處,否則將追究法律責任。

相關文章