翼度科技»论坛 编程开发 python 查看内容

.NET6+Quartz实现定时任务

8

主题

8

帖子

24

积分

新手上路

Rank: 1

积分
24
在实际工作中,经常会有一些需要定时操作的业务,如:定时发邮件,定时统计信息等内容,那么如何实现才能使得我们的项目整齐划一呢?本文通过一些简单的小例子,简述在.Net6+Quartz实现定时任务的一些基本操作,及相关知识介绍,仅供学习分享使用,如有不足之处,还请指正。
什么是定时任务?

定时任务,也叫任务调度,是指在一定的载体上,根据具体的触发规则,执行某些操作。所以定时任务需要满足三个条件:载体(Scheduler),触发规则(Trigger),具体业务操作(Job)。如下所示:

什么是Quartz?

Quartz 是一个开源的作业调度框架,它完全由 Java 写成,并设计用于 J2SE 和 J2EE 应用中。它提供了巨大的灵 活性而不牺牲简单性。你能够用它来为执行一个作业而创建简单的或复杂的调度。它有很多特征,如:数据库支持,集群,插件,EJB 作业预构 建,JavaMail 及其它,支持 cron-like 表达式等等。虽然Quartz最初是为Java编写的,但是目前已经有.Net版本的Quartz,所以在.Net中应用Quartz已经不再是奢望,而是轻而易举的事情了。
Github上开源网址为:https://github.com/quartznet

 
关于Quartz的快速入门和API文档,可以参考:https://www.quartz-scheduler.net/documentation/quartz-3.x/quick-start.html
 
涉及知识点

在Quartz框架中,主要接口和API如下所示:

 
 其中IScheduler,ITrigger , IJob 三者之间的关系,如下所示:

 
 
 Quartz安装

为了方便,本示例创建一个基于.Net6.0的控制台应用程序,在VS2022中,通过Nuget包管理器进行安装,如下所示:

 
创建一个简单的定时器任务

要开发一个简单,完整且能运行的定时器任务,步骤如下所示:
1. 创建工作单元Job

创建任务需要实现IJob接口,如下所示:
  1. 1 using Quartz;
  2. 2 using System.Diagnostics;
  3. 3
  4. 4 namespace DemoQuartz.QuartzA.Job
  5. 5 {
  6. 6     /// <summary>
  7. 7     /// 测试任务,实现IJob接口
  8. 8     /// </summary>
  9. 9     public class TestJob : IJob
  10. 10     {
  11. 11         public TestJob()
  12. 12         {
  13. 13             Console.WriteLine("执行构造函数");//表示每一次计划执行,都是一次新的实例
  14. 14         }
  15. 15
  16. 16         public Task Execute(IJobExecutionContext context)
  17. 17         {
  18. 18             return Task.Run(() =>
  19. 19              {
  20. 20                  Console.WriteLine($"******************************");
  21. 21                  Console.WriteLine($"测试信息{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}");
  22. 22                  Console.WriteLine($"******************************");
  23. 23                  Console.WriteLine();
  24. 24              });
  25. 25         }
  26. 26     }
  27. 27 }
复制代码
2. 创建时间轴Scheduler

时间轴也是任务执行的载体,可以通过StdSchedulerFactory进行获取,如下所示:
  1. 1 //创建计划单元(时间轴,载体)
  2. 2 StdSchedulerFactory schedulerFactory = new StdSchedulerFactory();
  3. 3 var scheduler = await schedulerFactory.GetScheduler();
  4. 4 await scheduler.Start();
复制代码
3. 创建触发规则Trigger

触发规则就是那些时间点执行任务,可通过TriggerBuilder进行构建,如下所示:
  1. 1 //Trigger时间触发机制
  2. 2 var trigger = TriggerBuilder.Create()
  3. 3     .WithIdentity("TestTrigger","TestGroup")
  4. 4     //.StartNow() //立即执行
  5. 5     .WithSimpleSchedule(w=>w.WithIntervalInSeconds(5).WithRepeatCount(5))//.RepeatForever()//无限循环
  6. 6     //.WithCronSchedule("5/10 * * * * ?") //通过Cron表达式定制时间触发规则, 示例表示从5开始,每隔10秒一次
  7. 7     .Build();
复制代码
4. 创建任务描述

任务描述定义了具体的任务名称,分组等内容。可通过JobBuilder进行构建,如下所示:
  1. 1 //Job详细描述
  2. 2 var jobDetail = JobBuilder.Create<TestJob>()
  3. 3     .WithDescription("这是一个测试Job")
  4. 4     .WithIdentity("TestJob", "TestGroup")
  5. 5     .Build();
复制代码
5. 建立三者联系

通过载体,将规则和工作单元串联起来,如下所示:
  1. 1 //把时间和任务通过载体关联起来
  2. 2 await scheduler.ScheduleJob(jobDetail, trigger);
复制代码
6. 简单示例测试

通过运行程序,示例结果如下所示:

 
传递参数

在Quartz框架下,如果需要给执行的Job传递参数,可以通过两种方式:
jobDetail.JobDataMap,工作描述时通过JobDataMap传递参数。
trigger.JobDataMap, 时间触发时通过JobDataMap传递参数。
在Job工作单元中,可以通过Context中对应的JobDataMap获取参数。
传递参数,如下所示:
  1. 1 //传递参数
  2. 2 jobDetail.JobDataMap.Add("name", "Alan");
  3. 3 jobDetail.JobDataMap.Add("age", 20);
  4. 4 jobDetail.JobDataMap.Add("sex", true);
  5. 5
  6. 6
  7. 7 //trigger同样可以传递参数
  8. 8 trigger.JobDataMap.Add("like1", "meimei");
  9. 9 trigger.JobDataMap.Add("like2", "football");
  10. 10 trigger.JobDataMap.Add("like3", "sing");
复制代码
获取参数,如下所示:
  1. 1 //获取参数
  2. 2 var name = context.JobDetail.JobDataMap.GetString("name");
  3. 3 var age = context.JobDetail.JobDataMap.GetInt("age");
  4. 4 var sex = context.JobDetail.JobDataMap.GetBoolean("sex") ? "男" : "女";
  5. 5
  6. 6 var like1 = context.Trigger.JobDataMap.GetString("like1");
  7. 7 var like2 = context.Trigger.JobDataMap.GetString("like2");
  8. 8 var like3 = context.Trigger.JobDataMap.GetString("like3");
  9. 9
  10. 10 //context.MergedJobDataMap.GetString("aa");//注意如果使用MergedJobDataMap,JobDetail和Trigger中用到相同的Key,则后面设置的会覆盖前面设置的。
复制代码
注意:如果使用MergedJobDataMap,JobDetail和Trigger中用到相同的Key,则后面设置的会覆盖前面设置的。
任务特性

假如我们的定时任务,执行一次需要耗时比较久,而且后一次执行需要等待前一次完成,并且需要前一次执行的结果作为参考,那么就需要设置任务的任性。因为默认情况下,工作单元在每一次运行都是一个新的实例,相互之间独立运行,互不干扰。所以如果需要存在一定的关联,就要设置任务的特性,主要有两个,如下所示:

  • [PersistJobDataAfterExecution]//在执行完成后,保留JobDataMap数据
  • [DisallowConcurrentExecution]//不允许并发执行,即必须等待上次完成后才能执行下一次
 
 
 以上两个特性,只需要标记在任务对应的类上即可。标记上后,只需要往对应的JobDataMap中添加值即可。
监听器

在Quartz框架下,有三种监听器,分别是:时间轴监听器ISchedulerListener,触发规则监听器ITriggerListener,任务监听器IJobListener。要实现对应监听器,实现对应接口即可。实现监听器步骤:
1. 创建监听器

根据不同的需要,可以创建不同的监听器,如下所示:
时间轴监听器SchedulerListener
  1.   1 public class TestSchedulerListener : ISchedulerListener
  2.   2 {
  3.   3     public Task JobAdded(IJobDetail jobDetail, CancellationToken cancellationToken = default)
  4.   4     {
  5.   5         return Task.Run(() => {
  6.   6             Console.WriteLine("Test Job is added.");
  7.   7         });
  8.   8     }
  9.   9
  10. 10     public Task JobDeleted(JobKey jobKey, CancellationToken cancellationToken = default)
  11. 11     {
  12. 12         return Task.Run(() => {
  13. 13             Console.WriteLine("Test Job is deleted.");
  14. 14         });
  15. 15     }
  16. 16
  17. 17     public Task JobInterrupted(JobKey jobKey, CancellationToken cancellationToken = default)
  18. 18     {
  19. 19         return Task.Run(() => {
  20. 20             Console.WriteLine("Test Job is Interrupted.");
  21. 21         });
  22. 22     }
  23. 23
  24. 24     public Task JobPaused(JobKey jobKey, CancellationToken cancellationToken = default)
  25. 25     {
  26. 26         return Task.Run(() => {
  27. 27             Console.WriteLine("Test Job is paused.");
  28. 28         });
  29. 29     }
  30. 30
  31. 31     public Task JobResumed(JobKey jobKey, CancellationToken cancellationToken = default)
  32. 32     {
  33. 33         return Task.Run(() => {
  34. 34             Console.WriteLine("Test Job is resumed.");
  35. 35         });
  36. 36     }
  37. 37
  38. 38     public Task JobScheduled(ITrigger trigger, CancellationToken cancellationToken = default)
  39. 39     {
  40. 40         return Task.Run(() => {
  41. 41             Console.WriteLine("Test Job is scheduled.");
  42. 42         });
  43. 43     }
  44. 44
  45. 45     public Task JobsPaused(string jobGroup, CancellationToken cancellationToken = default)
  46. 46     {
  47. 47         return Task.Run(() => {
  48. 48             Console.WriteLine("Test Jobs is paused.");
  49. 49         });
  50. 50     }
  51. 51
  52. 52     public Task JobsResumed(string jobGroup, CancellationToken cancellationToken = default)
  53. 53     {
  54. 54         return Task.Run(() => {
  55. 55             Console.WriteLine("Test Jobs is resumed.");
  56. 56         });
  57. 57     }
  58. 58
  59. 59     public Task JobUnscheduled(TriggerKey triggerKey, CancellationToken cancellationToken = default)
  60. 60     {
  61. 61         return Task.Run(() => {
  62. 62             Console.WriteLine("Test Jobs is un schedulered.");
  63. 63         });
  64. 64     }
  65. 65
  66. 66     public Task SchedulerError(string msg, SchedulerException cause, CancellationToken cancellationToken = default)
  67. 67     {
  68. 68         return Task.Run(() => {
  69. 69             Console.WriteLine("Test scheduler is error.");
  70. 70         });
  71. 71     }
  72. 72
  73. 73     public Task SchedulerInStandbyMode(CancellationToken cancellationToken = default)
  74. 74     {
  75. 75         return Task.Run(() => {
  76. 76             Console.WriteLine("Test scheduler is standby mode.");
  77. 77         });
  78. 78     }
  79. 79
  80. 80     public Task SchedulerShutdown(CancellationToken cancellationToken = default)
  81. 81     {
  82. 82         return Task.Run(() => {
  83. 83             Console.WriteLine("Test scheduler is shut down.");
  84. 84         });
  85. 85     }
  86. 86
  87. 87     public Task SchedulerShuttingdown(CancellationToken cancellationToken = default)
  88. 88     {
  89. 89         return Task.Run(() => {
  90. 90             Console.WriteLine("Test scheduler is shutting down.");
  91. 91         });
  92. 92     }
  93. 93
  94. 94     public Task SchedulerStarted(CancellationToken cancellationToken = default)
  95. 95     {
  96. 96         return Task.Run(() => {
  97. 97             Console.WriteLine("Test scheduleer is started.");
  98. 98         });
  99. 99     }
  100. 100
  101. 101     public Task SchedulerStarting(CancellationToken cancellationToken = default)
  102. 102     {
  103. 103         return Task.Run(() => {
  104. 104             Console.WriteLine("Test scheduler is starting.");
  105. 105         });
  106. 106     }
  107. 107
  108. 108     public Task SchedulingDataCleared(CancellationToken cancellationToken = default)
  109. 109     {
  110. 110         return Task.Run(() => {
  111. 111             Console.WriteLine("Test scheduling is cleared.");
  112. 112         });
  113. 113     }
  114. 114
  115. 115     public Task TriggerFinalized(ITrigger trigger, CancellationToken cancellationToken = default)
  116. 116     {
  117. 117         return Task.Run(() => {
  118. 118             Console.WriteLine("Test trigger is finalized.");
  119. 119         });
  120. 120     }
  121. 121
  122. 122     public Task TriggerPaused(TriggerKey triggerKey, CancellationToken cancellationToken = default)
  123. 123     {
  124. 124         return Task.Run(() => {
  125. 125             Console.WriteLine("Test trigger is paused.");
  126. 126         });
  127. 127     }
  128. 128
  129. 129     public Task TriggerResumed(TriggerKey triggerKey, CancellationToken cancellationToken = default)
  130. 130     {
  131. 131         return Task.Run(() => {
  132. 132             Console.WriteLine("Test trigger is resumed.");
  133. 133         });
  134. 134     }
  135. 135
  136. 136     public Task TriggersPaused(string? triggerGroup, CancellationToken cancellationToken = default)
  137. 137     {
  138. 138         return Task.Run(() => {
  139. 139             Console.WriteLine("Test triggers is paused.");
  140. 140         });
  141. 141     }
  142. 142
  143. 143     public Task TriggersResumed(string? triggerGroup, CancellationToken cancellationToken = default)
  144. 144     {
  145. 145         return Task.Run(() => {
  146. 146             Console.WriteLine("Test triggers is resumed.");
  147. 147         });
  148. 148     }
  149. 149 }
复制代码
触发规则监听器TriggerListener
  1. 1 /// <summary>
  2. 2 /// 触发器监听
  3. 3 /// </summary>
  4. 4 public class TestTriggerListener : ITriggerListener
  5. 5 {
  6. 6     public string Name => "TestTriggerListener";
  7. 7
  8. 8     public Task TriggerComplete(ITrigger trigger, IJobExecutionContext context, SchedulerInstruction triggerInstructionCode, CancellationToken cancellationToken = default)
  9. 9     {
  10. 10         //任务完成
  11. 11         return Task.Run(() => {
  12. 12             Console.WriteLine("Test trigger is complete.");
  13. 13         
  14. 14         });
  15. 15     }
  16. 16
  17. 17     public Task TriggerFired(ITrigger trigger, IJobExecutionContext context, CancellationToken cancellationToken = default)
  18. 18     {
  19. 19         return Task.Run(() => {
  20. 20             Console.WriteLine("Test trigger is fired.");
  21. 21
  22. 22         });
  23. 23     }
  24. 24
  25. 25     public Task TriggerMisfired(ITrigger trigger, CancellationToken cancellationToken = default)
  26. 26     {
  27. 27         return Task.Run(() => {
  28. 28             Console.WriteLine("Test trigger is misfired.");
  29. 29
  30. 30         });
  31. 31     }
  32. 32
  33. 33     public Task<bool> VetoJobExecution(ITrigger trigger, IJobExecutionContext context, CancellationToken cancellationToken = default)
  34. 34     {
  35. 35         return Task.Run(() => {
  36. 36             Console.WriteLine("Test trigger is veto.");
  37. 37             return false;//是否终止
  38. 38         });
  39. 39     }
  40. 40 }
复制代码
JobListener任务监听器
  1. 1 /// <summary>
  2. 2 /// TestJob监听器
  3. 3 /// </summary>
  4. 4 public class TestJobListener : IJobListener
  5. 5 {
  6. 6     public string Name => "TestJobListener";
  7. 7
  8. 8     public Task JobExecutionVetoed(IJobExecutionContext context, CancellationToken cancellationToken = default)
  9. 9     {
  10. 10         //任务被终止时
  11. 11         return Task.Run(() => {
  12. 12             Console.WriteLine("Test Job is vetoed.");
  13. 13         });
  14. 14     }
  15. 15
  16. 16     public Task JobToBeExecuted(IJobExecutionContext context, CancellationToken cancellationToken = default)
  17. 17     {
  18. 18         //任务被执行时
  19. 19         return Task.Run(() => {
  20. 20             Console.WriteLine("Test Job is to be executed.");
  21. 21         });
  22. 22     }
  23. 23
  24. 24     public Task JobWasExecuted(IJobExecutionContext context, JobExecutionException? jobException, CancellationToken cancellationToken = default)
  25. 25     {
  26. 26         //任务已经执行
  27. 27         return Task.Run(() => {
  28. 28             Console.WriteLine("Test Job was executed.");
  29. 29         });
  30. 30     }
  31. 31 }
复制代码
2. 添加监听

在时间轴上的监听管理器中进行添加,如下所示:
  1. 1 //增加监听
  2. 2 scheduler.ListenerManager.AddJobListener(new TestJobListener());
  3. 3 scheduler.ListenerManager.AddTriggerListener(new TestTriggerListener());
  4. 4 scheduler.ListenerManager.AddSchedulerListener(new TestSchedulerListener());
复制代码
日志管理

在Quartz框架中,创建之前会进行日志创建检测,所以如果需要获取框架中的日志信息,可以进行创建实现ILogProvider,如下所示:
  1. 1 public class TestLogProvider : ILogProvider
  2. 2 {
  3. 3     public Logger GetLogger(string name)
  4. 4     {
  5. 5         return (level, func, exception, parameters) =>
  6. 6         {
  7. 7             if (level >= Quartz.Logging.LogLevel.Info && func != null)
  8. 8             {
  9. 9                 Console.WriteLine("[" + DateTime.Now.ToLongTimeString() + "] [" + level + "] " + func(), parameters);
  10. 10             }
  11. 11             return true;
  12. 12         };
  13. 13     }
  14. 14
  15. 15     public IDisposable OpenMappedContext(string key, object value, bool destructure = false)
  16. 16     {
  17. 17         throw new NotImplementedException();
  18. 18     }
  19. 19
  20. 20     public IDisposable OpenNestedContext(string message)
  21. 21     {
  22. 22         throw new NotImplementedException();
  23. 23     }
  24. 24 }
复制代码
然后在当前的Scheduler中,添加日志即可,如下所示:
  1. 1 //日志
  2. 2 LogProvider.SetCurrentLogProvider(new TestLogProvider());
复制代码
完整示例

在添加了监听器,日志,参数传递,任务特性后,完整的目录结构,如下所示:

 
 示例截图

 
以上就是.Net6.0+Quartz开发控制台定时任务调度的全部内容。以上任务都是硬编码的固定程序,包括任务的启停,那么如果能通过可视化界面来创建以及管理任务,是不是一件很爽的事情呢,这也是后续需要探讨的内容。

来源:https://www.cnblogs.com/hsiang/archive/2023/03/26/17255074.html
免责声明:由于采集信息均来自互联网,如果侵犯了您的权益,请联系我们【E-Mail:cb@itdo.tech】 我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x

举报 回复 使用道具