13

介绍

我在ASP.Net框架4、webforms 网站上使用Quartz.Net 。基本上,用户应该能够手动触发一个批处理脚本,该脚本异步处理存储在数据库中的数千条记录。用户可以随时停止或暂停,调整一些变量,并在需要时继续处理(剩余记录)。 代码已完成并在本地工作(开发者机器、win7、vs2010、sql server express 2008 R2)。 它还在本地服务器上进行了测试(win server 2008 R2,sql server express 2008 R2)。它在两种环境下都可以正常工作,并使用所有预编译的代码进行了测试。问题是,一旦部署在远程服务器上

(win server 2008 R2),它实际上应该在其中运行(托管环境,不共享,不集群),它不能完全工作(详见下文)。调度程序被创建,但触发器,因此工作,不会触发

注意:我知道你们中的一些人会建议使用 Quartz 作为 Windows 服务,但尽管这样做有好处,但我真的很想知道为什么它不能作为嵌入式解决方案工作,因为它应该工作得很好喜欢在当地做)

细节

Quartz 2.1.2  
Common.Logging 2.1.2  
Common.Logging.NLog 2.0.0  
NLog 2.0.1.2

全球.asax

public static ISchedulerFactory SchedulerFactory;
public static IScheduler Scheduler;

void Application_Start(object sender, EventArgs e)
{
    SchedulerFactory = new StdSchedulerFactory();
    Scheduler = SchedulerFactory.GetScheduler();

    // Define a durable job instance (durable jobs can exist without triggers)
    IJobDetail job = JobBuilder.Create<MyJobClass>()
                                .WithIdentity("MyJob", "MyGroup")
                                .StoreDurably()
                                .Build();

    Scheduler.AddJob(job, false);
    Scheduler.Start();
}
void Application_End(object sender, EventArgs e)
{
    Scheduler.Shutdown(true);
}

process.aspx.cs(点击开始按钮)

// get records from DB, iterate, process, etc
...

IJobDetail job = ASP.global_asax.Scheduler.GetJobDetail(new JobKey("MyJob", "MyGroup"));
job.JobDataMap.Put("something1", 1);
job.JobDataMap.Put("something2", somevar);

ITrigger trigger = TriggerBuilder.Create()
                    .WithIdentity("MyTrigger", "MyGroup")
                    .StartNow()
                    .WithSimpleSchedule(x => x.WithIntervalInSeconds(5).RepeatForever())
                    .Build();

var triggersSet = new Quartz.Collection.HashSet<ITrigger> { trigger };

ASP.global_asax.Scheduler.ScheduleJob(job, triggersSet, true);

日志输出

本地日志

Default Quartz.NET properties loaded from embedded resource file  
Using default implementation for object serializer  
Using default implementation for ThreadExecutor  
Initialized Scheduler Signaller of type: Quartz.Core.SchedulerSignalerImpl  
Quartz Scheduler v.2.1.2.400 created.  
RAMJobStore initialized.  
Scheduler meta-data: Quartz Scheduler (v2.1.2.400) 'DefaultQuartzScheduler' with instanceId 'NON_CLUSTERED'   Scheduler class: 'Quartz.Core.QuartzScheduler' - running locally.   NOT STARTED.   Currently in standby mode.   Number of jobs executed: 0   Using thread pool 'Quartz.Simpl.SimpleThreadPool' - with 10 threads.   Using job-store 'Quartz.Simpl.RAMJobStore' - which does not support persistence. and is not clustered.  
Quartz scheduler 'DefaultQuartzScheduler' initialized  
Quartz scheduler version: 2.1.2.400  
Scheduler DefaultQuartzScheduler_$_NON_CLUSTERED started.  
Batch acquisition of 0 triggers  
Batch acquisition of 0 triggers

它继续记录0 触发器的批量获取,直到发生按钮单击:

Default Quartz.NET properties loaded from embedded resource file  
Batch acquisition of 1 triggers  
Producing instance of Job 'MyGroup.MyJob', class=MyJobClass  
Batch acquisition of 0 triggers  
Calling Execute on job MyGroup.MyJob  
Trigger instruction : NoInstruction  
Batch acquisition of 1 triggers  
Producing instance of Job 'MyGroup.MyJob', class=MyJobClass  
Batch acquisition of 0 triggers  
Calling Execute on job MyGroup.MyJob  
Trigger instruction : NoInstruction  
Batch acquisition of 1 triggers

部署日志

Default Quartz.NET properties loaded from embedded resource file  
Using default implementation for object serializer  
Using default implementation for ThreadExecutor  
Initialized Scheduler Signaller of type: Quartz.Core.SchedulerSignalerImpl  
Quartz Scheduler v.2.1.2.400 created.  
RAMJobStore initialized.  
Scheduler meta-data: Quartz Scheduler (v2.1.2.400) 'DefaultQuartzScheduler' with instanceId 'NON_CLUSTERED' Scheduler class: 'Quartz.Core.QuartzScheduler' - running locally. NOT STARTED. Currently in standby mode. Number of jobs executed: 0 Using thread pool 'Quartz.Simpl.SimpleThreadPool' - with 10 threads. Using job-store 'Quartz.Simpl.RAMJobStore' - which does not support persistence. and is not clustered.   
Quartz scheduler 'DefaultQuartzScheduler' initialized  
Quartz scheduler version: 2.1.2.400  
Scheduler DefaultQuartzScheduler_$_NON_CLUSTERED started.

这里保持这样。如您所见,与其他日志相比,它并没有尝试获取触发器(0触发器的行批量获取根本没有出现)。如果您仍然单击进程按钮,则日志会添加一行:

Default Quartz.NET properties loaded from embedded resource file

但没有其他事情发生。记录未处理(我知道,因为每次处理记录时,都会在数据库中标记)。不会发生错误,但不会触发触发器,也不会执行作业。此外,单击按钮时 CPU 使用率高达 50% 或更多,并且不会下降,除非您转到 IIS,停止并重新启动应用程序池。这种 CPU 消耗不会在本地发生。


更新 1

正如 LeftyX 所建议的那样,更改了单例调度程序的使用,但在远程服务器上仍然得到相同的行为。

更新 2

我还尝试使用ADOJobStore(而不是我使用的RAMJobStore)。现在它仍然可以在本地完美运行;但仍然不在线执行触发器(因此工作)。唯一的区别是在线 CPU 使用率不会达到 50%。现在我可以看到作业和触发器已创建(我查询表并查看这些记录是否存在),但从未被执行

4

4 回答 4

8

Quartz 没有任何问题,都是因为 IIS 应用程序池回收。我通过停止回收用于 Quartz 的池来修复错误:

  1. 转到IIS 管理器->应用程序池-> 创建一个新池,我将其命名为调度程序(任何名称都可以)
  2. 选择调度程序池->高级设置
  • 常规部分的启动模式中,选择AlwaysRunning (IIS 8.5) 或true for (IIS 7.5, 8)
  • Process Model Section-> Idle Timeout(minutes)设置为0(意思是:No Idel timeout)
  • 回收部分 ->定期时间间隔设置为0(意思是:不回收)
    3. 将您的 Quartz 站点部署到该应用程序池中。并向池发送一个请求以“唤醒您的应用程序”,它将一直运行直到您停止它。 而已。更新:使您的应用程序池始终处于活动状态的另一个解决方案是使用自动启动 ASP.NET 应用程序在此处输入图像描述

另一种选择:使用一些第三方 ping 工具(如uptimerobotdiy one)来保持每 x 秒(或分钟)刷新您的网站

于 2016-06-14T04:03:43.943 回答
4

我注意到的一件事是Scheduler在您的 asp.net 应用程序中使用。
您应该使用单例对象。

在你的process.aspx.cs这一行

IScheduler scheduler = new StdSchedulerFactory().GetScheduler();

创建一个新的调度程序,但您应该使用您在Application_Start.

如果您想访问单例实例,请在您的Global.asax.cs

 public static ISchedulerFactory SchedulerFactory;
 public static IScheduler Scheduler;

你可以在你的引用它process.aspx.cs

MvcApplication.Scheduler.ScheduleJob(job, triggersSet, true);

另一种解决方案是使用依赖注入。您可以在此处使用StructureMap此处Unity找到一些信息。

更新:

您可以在此处下载名为AspNet_Quartz的示例应用程序 (asp.net 4.0),并 此处查看它的工作原理。

于 2013-08-13T13:21:33.287 回答
4

问题与调度程序有关,IIS而不是与调度程序Quartz.NETHangfire有关。另一方面,网上发布了很多解决方法,但只有其中一些有效。在我看来,没有必要应用大量的配置设置。只需在您发布应用程序并享受的服务器上安装Keep Alive Service For IIS 6.0/7.5 。之后,您发布的应用程序将在应用程序池回收、IIS/应用程序重新启动等之后仍然存在。希望这有助于...

于 2015-11-12T13:07:38.967 回答
2

我刚刚遇到了一个可能会咬其他人的类似问题 - 在打开调试并收到“批量获取 0 个触发器”消息后,它让我想到了这个 SO 问题。
我的工作是每 2 小时一次,如下所示:

"0 0 0/2 * * ?"

我想让它更频繁,所以每 2 分钟这样一次:

"0 0/2 0 * ?"

我什至尝试了https://cronexpressiondescriptor.azurewebsites.net/这给了我一个很大的线索,我应该更仔细地阅读:每 2 小时,在每月的第 0 天, 这最终让我意识到我真正的意思是:

"0 0/2 * * ?"

所以教训是,当“左移”你的 cron 时,用 * 回填,而不是 0。

希望对其他人有所帮助。

于 2017-10-09T16:42:12.473 回答