在web程序上实现定时器,有一定难度,浏览器端其实只需要使用js的定时器就可以实现,但服务端如何实现呢?都是使用Global.asax加Timer 实现的,但很多文章却没有提到这种设计的问题。


 基本代码很简单:

Global.asax文件内容:

System.Timers.Timer timer = null;
    void Application_Start(object sender, EventArgs e) 
    {
        if (timer != null)
        {
            timer.Stop();
            timer.Close();
            timer = null;
        }
        int Interval = 1000 * 60 * 10;//十分钟
        timer = new System.Timers.Timer(Interval);//十分钟
        timer.Elapsed += new System.Timers.ElapsedEventHandler(Send);
        timer.Interval = Interval;
        timer.Enabled = true; 
        timer.Start();       
       
        DHC.EAS.Common.LogInfo.Info("当前的应用正被初始化......");
        // 在应用程序启动时运行的代码6
       // Castle.ActiveRecord.Framework.Config.XmlConfigurationSource source = new Castle.ActiveRecord.Framework.Config.XmlConfigurationSource("../../appconfig.xml");
        IConfigurationSource source = System.Configuration.ConfigurationManager.GetSection("activerecord") as IConfigurationSource;
        Castle.ActiveRecord.ActiveRecordStarter.Initialize(System.Reflection.Assembly.Load("Eas.Entity"), source);//可以针对任何在该程序集下的类  

        DHC.EAS.Common.LogInfo.Info("当前的应用初始化完毕.");
    }
    public void Send(object sender, System.Timers.ElapsedEventArgs e)
    {
        DHC.EAS.Bo.SendMsg.Send();       
    } 
  
    protected void Application_BeginRequest(Object sender, EventArgs e)
    {
        
    }     
    void Application_End(object sender, EventArgs e) 
    {
        //  在应用程序关闭时运行的代码
        DHC.EAS.Common.LogInfo.Info("当前的应用被关闭");
        new DHC.EAS.Common.AppException("当前的应用被关闭");
        if (timer != null)
        {
            timer.Stop();
            timer.Close();
            timer = null;
        }
    }
        
    void Application_Error(object sender, EventArgs e) 
    { 
        // 在出现未处理的错误时运行的代码
        Exception ex = Server.GetLastError().GetBaseException();  
        new DHC.EAS.Common.AppException("当前的应用发生错误",ex);
        //处理完及时清理异常 
        Server.ClearError();        
    }

    void Session_Start(object sender, EventArgs e) 
    {
        // 在新会话启动时运行的代码
        //DHC.EAS.Common.LogInfo.Info("回话时间间隔(分钟)" + Session.Timeout.ToString());
        //DHC.EAS.Common.LogInfo.Info("开始一个Session = " + Session.SessionID);
    }
    

    void Session_End(object sender, EventArgs e) 
    {
        // 在会话结束时运行的代码。 
        // 注意: 只有在 Web.config 文件中的 sessionstate 模式设置为
        // InProc 时,才会引发 Session_End 事件。如果会话模式设置为 StateServer 
        // 或 SQLServer,则不会引发该事件。
        //DHC.EAS.Common.LogInfo.Info("结束一个Session = " + Session.SessionID);

    }



其实这样做有两点需要考虑1 Application_Start必须发生,2 IIS可能进行回收,导致发生Application_End事件

1Application_Start必须发生

要求必须有用户访问网页,如果没有人访问网页,则Application_Start不会发生,timer不会启动,定时也就无从谈起。

2Application_End事件

例如发布了文件,修改了webconfig等都会导致Application_End事件,当然还有长时间没人访问,IIS的回收机制也会导致。然后timer也就不存在了,定时也就无从谈起。其实在这个之后可以通过代码访问自己的网站,保证Application_End事件发生后,Application_Start发生。

其实还有人提出使用HttpRuntime.Cache,就是添加一个变量,设定一个过期时间,在过期时通过回调函数调用需要的操作。这种方式能不能用,我没有测试,理论上可行,但个人感觉还是会存在上边的问题。

其实这种定时操作,我个人还是建议使用服务程序去处理。