运行计划任务的最佳方式

今天,我们已经构建了一个控制台应用程序,用于运行我们的ASP计划任务。网的网站。但是我认为这种方法容易出错,而且很难维护。如何执行计划任务(在windows/IIS/ASP。网络环境)

更新:

任务示例:

  • 从数据库中的电子邮件队列发送电子邮件
  • 从数据库中删除过时的对象
  • 从谷歌AdWords检索统计数据,并在数据库中填充一个表。
210935 次浏览

我不确定你指的是哪种计划任务。如果你的意思是“每小时刷新foo.xml”类型的任务,那么使用Windows的计划任务系统。(通过“at”命令,或者通过控制器。)让它运行一个控制台应用程序,或者请求一个特殊的页面来启动这个过程。

编辑:我应该补充,这是一个好的方式,让你的IIS应用程序运行在预定的点。因此,假设您希望每30分钟检查一次数据库,并通过电子邮件提醒用户一些数据,您可以使用计划任务来请求这个页面,从而让IIS处理事情。

如果你的需求更复杂,你可以考虑创建一个Windows服务,并让它运行一个循环来做你需要的任何处理。这样做还有一个好处,就是可以将代码分离出来用于扩展或管理目的。缺点是,您需要处理Windows服务。

这是Jeff Atwood为Stackoverflow开发的技术是我遇到的最简单的方法。它依赖于“已删除的缓存项”;回调机制内置到ASP。NET的缓存系统

更新:Stackoverflow已经超越了这种方法。它只在网站运行时起作用,但这是一个非常简单的技术,对许多人都有用。

还可以查看石英。网

我们也使用控制台应用程序。如果您使用像Log4net这样的日志工具,您可以正确地监视它们的执行。另外,我不确定它们为什么比网页更难维护,因为如果设计得当,两者之间可能会共享一些相同的代码库。

如果你不愿意让这些任务定时运行,你可以在你网站的管理部分设置一个网页作为队列。用户输入一个运行任务的请求,它反过来在MyProcessQueue表中插入一个空白的日期戳记录,您的计划任务每X分钟检查一次MyProcessQueue中的新记录。这样,它只在客户希望它运行时运行。

希望这些建议能有所帮助。

如果你拥有服务器,你应该使用windows任务调度程序。使用AT /?从命令行查看选项。

否则,在基于web的环境中,您可能不得不做一些讨厌的事情,比如设置一台不同的机器,以定时间隔向某个页面发出请求。

一种选择是设置一个windows服务,让它调用你的计划任务。

在winforms中,我使用了计时器,不认为这将在ASP中很好地工作。网

我所有的任务(这需要被安排)为一个网站保存在网站内,并从一个特殊的页面调用。然后我写了一个简单的Windows服务,它会经常调用这个页面。一旦页面运行,它就返回一个值。如果我知道有更多的工作要做,我会再次运行页面,立即,否则我会过一段时间再运行它。这对我来说工作得很好,并保持我所有的任务逻辑与web代码。在编写简单的Windows服务之前,我使用Windows调度器每x分钟调用一次页面。

另一种方便的方法是使用像Pingdom这样的监视服务。将他们的http检查指向运行您的服务代码的页面。让页面返回结果,然后可以使用该结果触发Pingdom,在出现问题时发送警报消息。

还有另一种方法:

1)创建一个“心跳”web脚本,负责在任务到期或过期时启动任务。

2)在某个地方创建一个预定的进程(最好是在同一个web服务器上),它会命中webscript,并强制它以定期的时间间隔运行。(例如,windows调度任务,使用IE或其他工具悄悄启动热节拍脚本)

任务代码包含在web脚本中的事实纯粹是为了保持代码 web应用程序代码库(假设两者相互依赖),这对web开发人员来说更容易管理。

另一种方法是创建一个可执行的服务器脚本/程序,它自己执行所有的计划工作,并将可执行文件本身作为计划任务运行。这可以实现web应用程序和计划任务之间的基本解耦。因此,如果你需要在web应用程序/数据库可能关闭或无法访问的情况下运行计划任务,你应该使用这种方法。

我已经在一个ASP中成功地使用了Abidar。NET项目(这里是一些背景信息)。

此方法的唯一问题是,如果ASP。NET web应用程序从内存中卸载。由于使用率低)。我尝试过的一件事是创建一个任务,每5分钟访问一次web应用程序,让它保持活跃,但这似乎并不可靠,所以现在我使用Windows调度器和基本的控制台应用程序来做这件事。

理想的解决方案是创建一个Windows服务,尽管这可能是不可能的。如果您正在使用共享托管环境)。从维护的角度来看,这也使得在web应用程序中保持内容更容易一些。

创建自定义Windows服务

我把一些关键任务设置为预定的控制台应用程序,发现它们很难维护。我创建了一个带有“心跳”的Windows服务,它每隔几分钟就会检查数据库中的日程安排。结果很好。

话虽如此,我仍然在使用预定的控制台应用程序来完成大部分非关键的维护任务。如果没坏,就别修。

我发现这对所有参与者来说都很简单:

  • 创建一个webservice方法,如DoSuchAndSuchProcess
  • 创建一个控制台应用程序来调用这个webmethod。
  • 在任务调度程序中调度控制台应用程序。

使用这种方法,所有的业务逻辑都包含在你的web应用程序中,但你有windows任务管理器的可靠性,或任何其他商业任务管理器来启动它并记录任何返回信息,如执行报告。使用web服务而不是发布到页面有一点优势,因为它更容易从web服务获取返回数据。

此外,如果你的应用程序使用SQL SERVER,你可以使用SQL代理来安排你的任务。这是我们通常放置重复出现的数据驱动代码(电子邮件提醒、定期维护、清洗等)的地方。SQL Agent内置的一个很好的特性是失败通知选项,它可以在关键任务失败时提醒您。

这个库工作起来很有魅力 http://www.codeproject.com/KB/cs/tsnewlib.aspx < / p >

它允许您通过. net代码直接管理Windows计划任务。

一个新的任务调度器类库

注意:自从这个库创建以来,微软已经为Windows Vista引入了一个新的任务调度器(task scheduler 2.0)。这个库是Task Scheduler 1.0接口的包装器,该接口在Vista中仍然可用,并且与Windows XP、Windows Server 2003和Windows 2000兼容。

http://www.codeproject.com/KB/cs/tsnewlib.aspx

你可以很容易地创建一个Windows服务,使用线程池间隔运行代码。RegisterWaitForSingleObject”方法。它真的很光滑,很容易安装。这种方法比使用框架中的任何计时器都更有效。

看看下面的链接,了解更多信息:

< p > 在。net中使用Windows服务运行周期进程: < br > http://allen-conway-dotnet.blogspot.com/2009/12/running-periodic-process-in-net-using.html < / p >

为什么要重新发明轮子,使用Threading和Timer类。

    protected void Application_Start()
{
Thread thread = new Thread(new ThreadStart(ThreadFunc));
thread.IsBackground = true;
thread.Name = "ThreadFunc";
thread.Start();
}


protected void ThreadFunc()
{
System.Timers.Timer t = new System.Timers.Timer();
t.Elapsed += new System.Timers.ElapsedEventHandler(TimerWorker);
t.Interval = 10000;
t.Enabled = true;
t.AutoReset = true;
t.Start();
}


protected void TimerWorker(object sender, System.Timers.ElapsedEventArgs e)
{
//work args
}

使用Windows调度器运行网页。

为了防止恶意用户或搜索引擎蜘蛛运行它,当您设置计划任务时,只需使用查询字符串调用web页面,即:mypage.aspx?从= scheduledtask

然后在页面加载中,简单地使用条件: 如果请求。查询string["from"] == "scheduledtask") { / / executetask } < / p >

这样,任何搜索引擎爬行器或恶意用户都无法执行您的计划任务。