定时任务框架Quartz.NET
1、介绍
Quartz.NET
是一个开源的作业调度框架,非常适合在平时的工作中,定时轮询数据库同步,定时邮件通知,定时处理数据等。 Quartz.NET
允许开发人员根据时间间隔来调度作业.
官方学习文档:http://www.quartz-scheduler.net/documentation/index.html
安装包:
1 2 3
| <ItemGroup> <PackageReference Include="Quartz" Version="3.9.0" /> </ItemGroup>
|
Quartz
主要有三部分组成任务(Job
)、触发器(Trigger
)和调度器(Schedule
)
Job
就是执行的作业,Job
需要实现IJob
接口,实现Execute
方法。Job
中执行的参数从Execute
方法的参数中获取。
触发器常用的有两种:SimpleTrigger
触发器和CronTrigger
触发器。
SimpleTrigger
触发器: 实现简单业务,如每隔几分钟,几小时触发执行,并限制执行次数。
CronTrigger
触发器:包含7个字段,秒 分 时 月内日期 月 周内日期 年(可选)
调度器就是将任务和触发器绑定,让触发器触发的时候去执行任务(调度器:负责创建,启动,停止和删除作业)。
工作流程:
1 2 3
| scheduler是quartz的独立运行容器,trigger和job都可以注册在scheduler容器中,一个job可以有多个触发器,而一个触发器只能属于一个job。
Quartz中有一个调度线程QuartzSchedulerThread,调度线程可以找到将要被触发的trigger和job,然后在ThreadPool中获取一个线程来执行这个job。
|
2、JobFactory
创建
创建job
工厂,在IOC
中放入Job
,然后再工厂中获取容器中的Job
对象。事实上,你实现工厂的核心就是定义**IJob**
实现类的实例化规则!
创建一个QuartzService
解决方案文件夹,在该文件夹中创建Cms.Quartz
项目。
并且,在该项目中按照相应的包。
创建JobFactory
类,实现IJobFactory
接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| public class JobFactory : IJobFactory { private readonly IServiceProvider serviceProvider; public JobFactory(IServiceProvider serviceProvider) { this.serviceProvider = serviceProvider; } public IJob NewJob(TriggerFiredBundle bundle, IScheduler scheduler) { var job = serviceProvider.GetService(bundle.JobDetail.JobType) as IJob; return job!; }
public void ReturnJob(IJob job) { var disposable = job as IDisposable; if (disposable != null) { disposable.Dispose(); } } }
|
3、创建Job
这个类必须要继承IJob
,实现具体的任务。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| namespace Cms.Quartz { [DisallowConcurrentExecution] public class QuartzJob : IJob {
private readonly MyDbContext myDbContext; public QuartzJob(MyDbContext dbContext) { this.myDbContext = dbContext; } public async Task Execute(IJobExecutionContext context) { Console.WriteLine("具体执行的任务,统计用户注册数"); var list = await myDbContext.Users.Where(u => u.DelFlag == false).ToListAsync(); Console.WriteLine("userCount=" + list.Count());
var jobDetails = context.JobDetail; var trigger = context.Trigger; JobDataMap dataMap = context.JobDetail.JobDataMap;
Console.WriteLine($"用户名是: {dataMap.GetString("UserName")}"); Console.WriteLine($"JobKey:{jobDetails.Key},Group:{jobDetails.Key.Group}\r\n" + $"Trigger:{trigger.Key}\r\n" + $"RunTime:{context.JobRunTime}" + $"ExecuteTime:{DateTime.Now}");
} }
|
4、创建调度器
创建一个调度器的接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
| namespace Cms.Quartz { public interface IControllerCenter { Task Start(); Task RunJob(); Task PauseJob(); Task ResumeJob(); } }
|
实现上面的接口
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
| namespace Cms.Quartz { public class ControllerCenter : IControllerCenter { private readonly IJobFactory jobFactory; private Task<IScheduler> scheduler; public ControllerCenter(IJobFactory jobFactory) { this.jobFactory = jobFactory; scheduler = GetScheduler(); scheduler.Result.JobFactory = jobFactory; }
private Task<IScheduler> GetScheduler() { if (scheduler != null) return scheduler; else {
var properties = new NameValueCollection { ["quartz.serializer.type"] = "binary" }; ISchedulerFactory sf = new StdSchedulerFactory(properties); this.scheduler = sf.GetScheduler(); return scheduler; } } public async Task PauseJob() { var jobKey = new JobKey("job1", "group1"); if (await this.scheduler.Result.CheckExists(jobKey)) { await this.scheduler.Result.PauseJob(jobKey); Console.WriteLine("PauseJob Success!"); } else { Console.WriteLine("Not IsExists JobKey"); } } public async Task ResumeJob() { var jobKey = new JobKey("job1", "group1"); if (await this.scheduler.Result.CheckExists(jobKey)) { await this.scheduler.Result.ResumeJob(jobKey); Console.WriteLine("ResumeJob Success!"); } else { Console.WriteLine("Not IsExists JobKey"); } } public async Task RunJob() { var jobKey = new JobKey("job1", "group1"); if (await this.scheduler.Result.CheckExists(jobKey)) { Console.WriteLine("JobKey Exists"); } else { Console.WriteLine("JobKey Allow"); if (!this.scheduler.Result.IsStarted) { Console.WriteLine("quartz is not started"); await this.Start(); } var job = JobBuilder.Create<QuartzJob>() .UsingJobData("UserName", "zhangsan") .WithIdentity("job1", "group1") .Build(); var trigger = TriggerBuilder.Create() .WithIdentity("trigger1", "group1") .StartNow() .WithSimpleSchedule(a => { a.RepeatForever(); a.WithIntervalInSeconds(3); }) .ForJob(job) .Build(); await this.scheduler.Result.ScheduleJob(job, trigger); } } public async Task Start() { try { if (this.scheduler.Result.IsStarted) { Console.WriteLine("quartz is started"); } else { Console.WriteLine("quartz start!"); await this.scheduler.Result.Start(); } } catch (Exception ex) { Console.WriteLine(ex.Message); } } } }
|
5、添加服务
在Cms.Web
这个Web
项目中,添加服务。
1 2 3 4 5 6 7 8 9
| builder.Services.AddSingleton<MailQueueManager>();
builder.Services.AddSingleton<IJobFactory, JobFactory>();
builder.Services.AddSingleton<IControllerCenter, ControllerCenter>();
builder.Services.AddSingleton<QuartzJob>();
|
6、开启任务
创建QuartzController.cs
控制器
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| public class QuartzController : Controller { private readonly IControllerCenter _controllerCenter; public QuartzController(IControllerCenter controllerCenter) { this._controllerCenter = controllerCenter; } [HttpGet] public async Task<IActionResult> Start() { await _controllerCenter.Start(); return Content("开启任务调度"); }
[HttpGet] public async Task<IActionResult> Run() { await _controllerCenter.RunJob(); return Content("开始运行JOB"); ; }
public IActionResult Index() { return View(); } }
|
先执行Start
方法,然后再执行Run
方法。
可以在系统的左侧添加一个菜单项,在对应的页面中添加相应的按钮来开启任务调度与开启JOB
运行。
参考文档:https://www.cnblogs.com/qtiger/p/13633965.html
该文档中有关于CronTrigger
触发器的详细说明