方法一
using System;
using SystemData;
using SystemConfiguration;
using SystemCollections;
using SystemWeb;
using SystemWebSecurity;
using SystemWebSessionState;
using SystemTimers;
using SystemNet;
using SystemIO;
using SystemText;
using SystemThreading;
namespace
{
public class Global : SystemWebHttpApplication
{
protected void Application_Start(object sender EventArgs e)
{
//定义定时器
SystemTimersTimer myTimer = new SystemTimersTimer();
myTimerElapsed += new ElapsedEventHandler(myTimer_Elapsed);
myTimerEnabled = true;
myTimerAutoReset = true;
}
void myTimer_Elapsed(object source ElapsedEventArgs e)
{
try
{
LogSaveNote(DateTimeNowToString(yyyyMMdd HH:mm:ss) + :AutoTask is Working!);
YourTask();
}
catch (Exception ee)
{
LogSaveException(ee);
}
}
void YourTask()
{
//在这里写你需要执行的任务
}
protected void Application_End(object sender EventArgs e)
{
LogSaveNote(DateTimeNowToString(yyyyMMdd HH:mm:ss) + :Application End!);
//下面的代码是关键可解决IIS应用程序池自动回收的问题
ThreadSleep();
//这里设置你的web地址可以随便指向你的任意一个aspx页面甚至不存在的页面目的是要激发Application_Start
string url =
HttpWebRequest myHttpWebRequest = (HttpWebRequest)WebRequestCreate(url);
HttpWebResponse myHttpWebResponse = (HttpWebResponse)myHttpWebRequestGetResponse();
Stream receiveStream = myHttpWebResponseGetResponseStream();//得到回写的字节流
}
}
}
原理Globalasax 可以是中应用程序或会话事件处理程序我们用到了Application_Start(应用程序开始事件)和Application_End(应用程序结束事件)当应用程序开始时启动一个定时器用来定时执行任务YourTask()方法这个方法里面可以写上需要调用的逻辑代码可以是单线程和多线程当应用程序结束时如IIS的应用程序池回收让去访问当前的这个web地址这里需要访问一个aspx页面这样就可以重新激活应用程序Log类是一个记录日志的一个类下面是测试时生成的日志信息
================================================================
:::AutoTask is Working!
:::AutoTask is Working!
:::AutoTask is Working!
:::Application End!
:::AutoTask is Working!
:::AutoTask is Working!
从日志中发现当手动回收IIS的应用程序池之后计划任务还在执行说明我们的目的达到了
如果将Application_End中的代码注释掉会发现Application End之后计划任务停止工作了如下
================================================================
:::AutoTask is Working!
:::AutoTask is Working!
:::AutoTask is Working!
:::Application End!
局限性可以解决应用程序池自动或者手动回收但是无法解决IIS重启或者web服务器重启的问题当然这种情况出现的时候不多而且如果有人访问你的网站的时候又会自动激活计划任务了
方案二
<%@ Application Language=C# %>
<%@ import Namespace=SystemIO %>
<script runat=server>
void Application_Start(object sender EventArgs e)
{
// 在应用程序启动时运行的代码
SystemTimersTimer myTimer = new SystemTimersTimer();
myTimerElapsed += new SystemTimersElapsedEventHandler(OnTimedEvent);
myTimerInterval = ;
myTimerEnabled = 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)
{
// 在会话结束时运行的代码
// 注意: 只有在 nfig 文件中的 sessionstate 模式设置为
// InProc 时才会引发 Session_End 事件如果会话模式设置为 StateServer
// 或 SQLServer则不会引发该事件
}
private static void OnTimedEvent(object source SystemTimersElapsedEventArgs e)
{
//间隔时间执行某动作
//指定日志文件的目录
string fileLogPath = AppDomainCurrentDomainBaseDirectory + SystemLog;
string fileLogName = SoftPrj_CN_ + DateTimeNowToLongDateString() + _logtxt;
//定义文件信息对象
FileInfo finfo = new FileInfo(fileLogPath + fileLogName);
//创建只写文件流
using (FileStream fs = finfoOpenWrite())
{
//根据上面创建的文件流创建写数据流
StreamWriter strwriter = new StreamWriter(fs);
//设置写数据流的起始位置为文件流的末尾
strwriterBaseStreamSeek( SeekOriginEnd);
//写入错误发生时间
strwriterWriteLine(发生时间: + DateTimeNowToString());
//写入日志内容并换行
//strwriterWriteLine(错误内容: + message);
strwriterWriteLine(错误内容: );
//写入间隔符
strwriterWriteLine();
strwriterWriteLine();
//清空缓沖区内容并把缓沖区内容写入基础流
strwriterFlush();
//关闭写数据流
strwriterClose();
fsClose();
}
}
</script>
方案三
<%@ Application Language=C# %>
<%@ Import Namespace=SystemIO %>
<%@ Import Namespace=SystemThreading %>
<script RunAt=server>
string LogPath;
Thread thread;
void WriteLog()
{
while (true)
{
StreamWriter sw = new StreamWriter(LogPath true EncodingUTF);
swWriteLine(threadName + : + DateTimeNowToString());
swClose();
ThreadCurrentThreadJoin( * );//阻止秒
}
}
void Application_Start(object sender EventArgs e)
{
LogPath = HttpContextCurrentServerMapPath(logtxt); //在应用程序启动时运行的代码
thread = new Thread(new ThreadStart(WriteLog));
threadName = 写登录日志线程;
threadStart();
}
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)
{
// 在会话结束时运行的代码
// 注意: 只有在 nfig 文件中的 sessionstate 模式设置为
// InProc 时才会引发 Session_End 事件如果会话模式设置为 StateServer
// 或 SQLServer则不会引发该事件
}
</script>