一、介绍 Quartz.Net是根据Java的Quartz用C#改写而来,最新的版本是3.0.6,源码在https://github.com/quartznet/quartznet 。主要作用是做一些周期性的工作,或者定时工作。比如每天凌晨2点对前一天的数据统计。
二、简单的案例 以WebApi项目举例,用VS脚手架功能新建WebApi项目。
1 2 3 4 5 public void ConfigureServices (IServiceCollection services ){ services.AddMvc(); services.AddSingleton<ISchedulerFactory, StdSchedulerFactory>(); }
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 [Route("api/[controller]" ) ] public class ValuesController : Controller { private readonly ISchedulerFactory _schedulerFactory; private IScheduler _scheduler; public ValuesController (ISchedulerFactory schedulerFactory ) { this ._schedulerFactory = schedulerFactory; } [HttpGet ] public async Task<string []> Get() { _scheduler = await _schedulerFactory.GetScheduler(); await _scheduler.Start(); var trigger = TriggerBuilder.Create() .WithSimpleSchedule(x => x.WithIntervalInSeconds(2 ).RepeatForever()) .Build(); var jobDetail = JobBuilder.Create<MyJob>() .WithIdentity("job" , "group" ) .Build(); await _scheduler.ScheduleJob(jobDetail, trigger); return await Task.FromResult(new string [] { "value1" , "value2" }); } }
1 2 3 4 5 6 7 8 9 10 11 12 13 public class MyJob : IJob { public Task Execute (IJobExecutionContext context ) { return Task.Run(() => { using (StreamWriter sw = new StreamWriter(@"C:\Users\Administrator\Desktop\error.log" , true , Encoding.UTF8)) { sw.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH-mm-ss" )); } }); } }
1 输出的结果: 2018-08-03 00-03-19 2018-08-03 00-03-20 2018-08-03 00-03-22 2018-08-03 00-03-24 2018-08-03 00-03-26
上面这种执行的Job没有参数,当需要参数可以通过下面两种方法传递参数:
1、在Trigger中添加参数值
1 2 3 4 5 6 var trigger3 = TriggerBuilder.Create() .WithSimpleSchedule(x =>x.WithIntervalInSeconds(2 ).RepeatForever()) .UsingJobData("key1" , 321 ) .UsingJobData("key2" , "123" ) .WithIdentity("trigger2" , "group1" ) .Build();
2、在Job中添加参数值
1 2 3 4 5 IJobDetail job = JobBuilder.Create<MyJob>() .UsingJobData("key1" , 123 ) .UsingJobData("key2" , "123" ) .WithIdentity("job1" , "group1" ) .Build();
通过下面方法在Job中获取参数值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 public class MyJob : IJob { public Task Execute (IJobExecutionContext context ) { var jobData = context.JobDetail.JobDataMap; var triggerData = context.Trigger.JobDataMap; var data = context.MergedJobDataMap; var value1= jobData.GetInt("key1" ); var value2= jobData.GetString("key2" ); var dateString = DateTime.Now.ToString("yyyy-MM-dd HH-mm-ss" );return Task.Run(() => { using (StreamWriter sw = new StreamWriter(@"C:\Users\Administrator\Desktop\error.log" , true , Encoding.UTF8)) { sw.WriteLine(dateString); } }); } }
当Job中的参数和Trigger中的参数名称一样时,用 context.MergedJobDataMap获取参数时,Trigger中的值会覆盖Job中的值。
3、上面那种情况只能适应那种,参数值不变的情况。假如有这种情况,这次的参数值是上一次执行后计算的值,就不能使用上面方法了。如 每两秒实现累加一操作,现在初始值是0,如果按照上面那种获取值的操作,一直都是0+1,返回值一直都是1。为了满足这个情况,只需要加一个特性[PersistJobDataAfterExecution]。
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 [PersistJobDataAfterExecution ] public class MyJob : IJob { public Task Execute (IJobExecutionContext context ) { var jobData = context.JobDetail.JobDataMap; var triggerData = context.Trigger.JobDataMap; var data = context.MergedJobDataMap; var value1 = jobData.GetInt("key1" ); var value2 = jobData.GetString("key2" ); var value3 = data.GetString("key2" ); var dateString = DateTime.Now.ToString("yyyy-MM-dd HH-mm-ss" ); Random random = new Random(); jobData["key1" ] = random.Next(1 , 20 ); jobData["key2" ] = dateString; return Task.Run(() => { using (StreamWriter sw = new StreamWriter(@"C:\Users\Administrator\Desktop\error.log" , true , Encoding.UTF8)) { sw.WriteLine($"{dateString} value1:{value1} value2:{value2} " ); } }); } }
三、Quartz.Net组成 Quartz主要有三部分组成任务(Job)、触发器(Trigger)和调度器(Schedule)。
3.1 任务
Job就是执行的作业,Job需要继承IJob接口,实现Execute方法。Job中执行的参数从Execute方法的参数中获取。
3.2 触发器
触发器常用的有两种:SimpleTrigger触发器和CronTrigger触发器。
SimpleTrigger:能是实现简单业务,如每隔几分钟,几小时触发执行,并限制执行次数。
1 2 3 4 5 var trigger = TriggerBuilder.Create() .WithSimpleSchedule(x => x.WithIntervalInSeconds(2 ).WithRepeatCount(5 )) .UsingJobData("key1" , 321 ) .WithIdentity("trigger" , "group" ) .Build();
CronTrigger:Cron表达式包含7个字段,秒 分 时 月内日期 月 周内日期 年(可选)。
举例:
1 2 3 4 5 6 var trigger = TriggerBuilder.Create() .WithCronSchedule("0 0 0 1 1 ?" ) .UsingJobData("key1" , 321 ) .UsingJobData("key2" , "trigger-key2" ) .WithIdentity("trigger4" , "group14" ) .Build();
“0 15 10 * * ? *” 每天上午10:15触发
“0 0-5 14 * * ?” 每天下午2点到下午2:05期间的每1分钟触发
3.3 调度器
调度器就是将任务和触发器绑定,让触发器触发的时候去执行任务。
https://www.cnblogs.com/MicroHeart/p/9402731.html
https://www.cnblogs.com/Leo_wl/p/8524600.html