ASP.NET 定时执行任务 global.asax

时间:2022-02-10 07:59:54

    方法一、  
     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>