根据不同需求,在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服务器重启的问题,当然这种情况出现的时候不多,而且如果有人访问你的网站的时候,又会自动激活计划任务了。
方法二:
<%@ "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> |
方法三:
<%@ "C#" %> <%@ "System.IO" %> <%@ "System.Threading" %> <script "server" > string LogPath; Thread void WriteLog() { while ( true ) { StreamWriter new StreamWriter(LogPath, true , Encoding.UTF8); sw.WriteLine(thread.Name ":" + DateTime.Now.ToString()); sw.Close(); Thread.CurrentThread.Join(1000 //阻止10秒 } } void Application_Start( object sender, EventArgs e) { LogPath "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) { // // // // } </script> |