根据不同需求,在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服务器重启的问题,当然这种情况出现的时候不多,而且如果有人访问你的网站的时候,又会自动激活计划任务了。
方法二:
<%@ 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>
|