最佳实践

JobDataMap
  1. 建议只存储基本数据(含String),避免序列化问题

  2. 作业执行期间,JobDetail和Trgger的底层共用一个JobDataMap 实例,因此Trigger的数据会覆盖Job中相同key的值。

  3. 每个独立触发器的JobDataMap 是独立的

  4. 在作业执行期间,建议使用MergedJobDataMap去检索key

    // 不建议
    var badMethod = context.JobDetail.JobDataMap.GetString("a-value");
    var alsoBadMethod = context.Trigger.JobDataMap.GetString("a-value");
    // 建议
    var goodMethod = context.MergedJobDataMap.GetString("a-value");
Job

​ 为统一作业的名称和群组,建议在IJob实现时就声明一个静态JobKey,便于触发器赋值使用

public class SomeJob : IJob
{
// 定义当前作业的名称和群组
public static readonly JobKey Key = new JobKey("job-name", "group-name"); public Task Execute(IJobExecutionContext context) { /* elided */ }
}
// 创建触发器
public async Task DoSomething(IScheduler schedule, CancellationToken ct)
{
var trigger = TriggerBuilder.Create()
.WithIdentity("a-trigger", "a-group")
.ForJob(SomeJob.Key)
.StartNow()
.Build(); await schedule.ScheduleJob(trigger, ct)
}
// 设置当前作业的触发器时
public async Task DoSomething(IScheduler schedule, CancellationToken ct)
{
await schedule.TriggerJob(SomeJob.Key, ct)
}
Trigger

​ 使用TriggerUtils进行触发器时间上的操作

  1. 根据触发器获得执行时间列表

    static IReadOnlyList ComputeFireTimes(IOperableTrigger trigg, ICalendar? cal, int numTimes)

    // 设置时间
    DateTimeOffset startCalendar = DateBuilder.DateOf(9, 30, 17, 1, 1, 2023);
    // 设置日期的触发器,间隔90天执行一次,从2023-01-01 17:30:09开始
    var dailyTrigger = new CalendarIntervalTriggerImpl
    {
    StartTimeUtc = startCalendar,
    RepeatIntervalUnit = IntervalUnit.Day,
    RepeatInterval = 90 // every ninety days
    };
    // 2023年1月1日加上360天 4*90 四个90天的循环
    DateTimeOffset targetCalendar = startCalendar.AddDays(360);
    // 获得从0开始的6次循环(0-5)
    var fireTimes = TriggerUtils.ComputeFireTimes(dailyTrigger, null, 6);
    // 获得第五次的循环日期,2023-01-01 17:30:09当天也会执行一次
    DateTimeOffset fifthTime = fireTimes[4]; // get the fifth fire time
    // 两次获得的日期相同
    Assert.AreEqual(targetCalendar, fifthTime, "Day increment result not as expected.");
  2. 获得触发器最后一次执行的时间

    static DateTimeOffset? ComputeEndTimeToAllowParticularNumberOfFirings(IOperableTrigger trigger, ICalendar? calendar, int numberOfTimes)

    // 设置时间
    DateTimeOffset startCalendar = DateBuilder.DateOf(9, 30, 17, 1, 1, 2023);
    // 设置日期的触发器,间隔90天执行一次,从2023-01-01 17:30:09开始
    var dailyTrigger = new CalendarIntervalTriggerImpl
    {
    StartTimeUtc = startCalendar,
    RepeatIntervalUnit = IntervalUnit.Day,
    RepeatInterval = 90 // every ninety days
    };
    // 2023年1月1日加上360天 4*90 四个90天的循环
    DateTimeOffset targetCalendar = startCalendar.AddDays(360);
    // 获得从0开始的4次循环(0-3),然后返回第四次执行的时间
    var endFireTimes = TriggerUtils.ComputeFireTimes(dailyTrigger, null, 4);
    // 两次获得的日期相同
    Assert.AreEqual(targetCalendar, fifthTime, "Day increment result not as expected.");
  3. 在两个日期之间触发器执行的时间列表

    static IReadOnlyList ComputeFireTimesBetween(IOperableTrigger trigg, ICalendar? cal, DateTimeOffset from, DateTimeOffset to)

    // 设置触发器
    var trigger = (IOperableTrigger) TriggerBuilder.Create()
    .WithDailyTimeIntervalSchedule(x => x
    .InTimeZone(TZConvert.GetTimeZoneInfo("GTB Standard Time"))
    // 每天从0点开始
    .StartingDailyAt(new TimeOfDay(0, 0, 0))
    // 跳过21点,第二次循环跳过了20和21点
    .EndingDailyAt(new TimeOfDay(22, 0, 0))
    // 15分钟循环一次
    .WithInterval(15, IntervalUnit.Minute)
    // 触发失效什么也不做
    .WithMisfireHandlingInstructionDoNothing()
    )
    .Build();
    var from = new DateTimeOffset(2023, 1, 31, 0, 0, 0, TimeSpan.Zero);
    var to = new DateTimeOffset(2023, 2, 2, 0, 0, 0, TimeSpan.Zero);
    //理论上会执行 ((23个小时+22个小时)*60分钟)/(15分钟/次) - 1次 = 179次
    var times = TriggerUtils.ComputeFireTimesBetween(trigger, null, from, to);
    // 小于200次
    Assert.That(times.Count, Is.LessThan(200));
ADO.NET JobStore
  1. 使用Quartz.NET的API写入数据库,不要直接操作数据库表
  2. 死锁
    • 触发器中的作业未被正确执行
  3. 集群之间的问题
    • 非集群与集群不能共用一个数据库
  4. 数据源连接大小
    • 建议将数据源最大连接大小配置为至少为线程池中的工作线程数加上三
    • 如果使用Api则需要额外的连接
Daylight Savings Time

​ 夏令时-中国在执行了6年就取消了,因此该最佳实现目前不适用于无夏令时的地区。

​ 因为SimpleTriggers为精确到毫秒,因此不受夏令时的影响;CronTriggers在转换的时候会受到夏令时的影响,可能会多触发一次;CalenderIntervalTrigger也会受影响,会导致偏移一小时的误差。

Jobs
  • 长时间运行的作业会阻止其他线程运行(例如已运行作业数量与线程池数量相等)
  • 不建议在作业中使用Thread.Sleep(),因为他会让线程一直被占用,建议在使用Sleep的位置重新安排自身或者其他作业来替代等待(通常等待是因为条件不为true,因此可以将该判断作为作业执行,周期短)
  • 因为作业是周期性的,因此当作业异常时会一直触发异常,此时异常过多会导致应用程序不稳定。建议在catch处解决异常或重新开启新的作业来消解异常的发生。
Listener
  • 不建议监听中执行大量工作,因为作业更合适做这些工作,监听只是监听。
  • 同Jobs的异常处理,异常多了容易崩溃。
对外调用
  • 不建议对外暴露,对于本机过于危险(例如NativeJob和SendEmailJob都可以用于恶意的目的)

Quartz.Net 官方教程 Best Practices的更多相关文章

  1. Unity性能优化(3)-官方教程Optimizing garbage collection in Unity games翻译

    本文是Unity官方教程,性能优化系列的第三篇<Optimizing garbage collection in Unity games>的翻译. 相关文章: Unity性能优化(1)-官 ...

  2. Unity性能优化(4)-官方教程Optimizing graphics rendering in Unity games翻译

    本文是Unity官方教程,性能优化系列的第四篇<Optimizing graphics rendering in Unity games>的翻译. 相关文章: Unity性能优化(1)-官 ...

  3. Quartz.net官方开发指南系列篇

    Quartz.NET是一个开源的作业调度框架,是OpenSymphony 的 Quartz API的.NET移植,它用C#写成,可用于winform和asp.net应用中.它提供了巨大的灵活性而不牺牲 ...

  4. Unity性能优化(2)-官方教程Diagnosing performance problems using the Profiler window翻译

    本文是Unity官方教程,性能优化系列的第二篇<Diagnosing performance problems using the Profiler window>的简单翻译. 相关文章: ...

  5. Unity性能优化(1)-官方教程The Profiler window翻译

    本文是Unity官方教程,性能优化系列的第一篇<The Profiler window>的简单翻译. 相关文章: Unity性能优化(1)-官方教程The Profiler window翻 ...

  6. jeecg表单页面控件权限设置(请先看官方教程,如果能看懂就不用看这里了)

    只是把看了官方教程后,觉得不清楚地方补充说明一下: 1. 2. 3. 4.用"jeecgDemoController.do?addorupdate"这个路径测试,不出意外现在应该可 ...

  7. [转]Google Guava官方教程(中文版)

    Google Guava官方教程(中文版) http://ifeve.com/google-guava/

  8. Google Guava官方教程(中文版)

    Google Guava官方教程(中文版) 原文链接  译文链接 译者: 沈义扬,罗立树,何一昕,武祖  校对:方腾飞 引言 Guava工程包含了若干被Google的 Java项目广泛依赖 的核心库, ...

  9. OpenGL官方教程——着色器语言概述

    OpenGL官方教程——着色器语言概述 OpenGL官方教程——着色器语言概述 可编程图形硬件管线(流水线) 可编程顶点处理器 可编程几何处理器 可编程片元处理器 语言 可编程图形硬件管线(流水线) ...

  10. [苏飞开发助手V1.0测试版]官方教程与升级报告

           [苏飞开发助手V1.0测试版]官方教程与升级报告导读部分----------------------------------------------------------------- ...

随机推荐

  1. UBOOT编译--- make xxx_deconfig过程详解(一)

    make xxx_deconfig过程详解 1. 前言 2. 概述 3. build变量的定义 4. 目标%config的定义 4.1 依赖 scripts_basic 4.1.1 语句$(if $ ...

  2. 在博客中实现播放音乐功能(QQ,网易,酷狗,虾米,百度)

    1.在页头head标签里添加: <link rel="stylesheet" href="https://static.likepoems.com/cdn/apla ...

  3. 手记系列之二 ----- 关于IDEA的一些使用方法经验

    前言 本篇文章主要介绍的关于本人在使用IDEA的一些使用方法,一些常用设置,一些插件推荐和使用.请注意,本文特长,2w多字加上几十张图片,建议收藏观看~ 前提准备 idea官网: https://ww ...

  4. Xmake v2.7.3 发布,包组件和 C++ 模块增量构建支持

    Xmake 是一个基于 Lua 的轻量级跨平台构建工具. 它非常的轻量,没有任何依赖,因为它内置了 Lua 运行时. 它使用 xmake.lua 维护项目构建,相比 makefile/CMakeLis ...

  5. AI音乐创作,让每一个人都成为音乐家

    从录音带.MP3到专业的耳机.音箱,随着音乐消费方式的不断升级,音乐创作的专业"门槛"也在AI技术的加持下逐渐大众化,创作者的创新设计.创作频率也在持续增强,能降低创作门槛且智能化 ...

  6. 第2-3-2章 环境搭建-文件存储服务系统-nginx/fastDFS/minio/阿里云oss/七牛云oss

    目录 5. 文件服务开发 5.1 环境搭建 5.1.1 数据库环境搭建 5.1.2 Nacos环境搭建 5.1.3 Nginx环境搭建 5.1.4 maven工程环境搭建 5. 文件服务开发 全套代码 ...

  7. 树莓派(香橙派)通过.NET IoT 操作SPI编写屏幕驱动 顺手做个四足机器人(一)

    摘要 这片文章主要是记录自己的整活过程,涉及到的技术包括.NET IoT, .NET Web, .NET MAUI,框架采用的也是最新的.NET 7. 本人是用的树莓派Zero 2 W(ubuntu- ...

  8. Python-OpenCV的安装及学习资料

    Conda环境安装 OpenCV pip install opencv-python opencv-contrib-python -i https://mirrors.aliyun.com/pypi/ ...

  9. VsCode搭建一个React项目

    安装Node.js 使用 npm -v检查安装成功 目前的 node 中都会自带 npm 所以不需要重新下载 直接切换至淘宝镜像即可 1.临时使用 :npm --registry https://re ...

  10. CheckBox 单选实现及取值

    <input name="ck" type="checkbox" value="1"/><span>按计划进行< ...