Quartz.Net 组件的封装使用

Quartz.Net是面向.NET的一款功能齐全的开源作业调度组件,你可以把它嵌入你的系统中实现作业调度,也可以基于Quartz.Net开发一套完整的作业调度系统。它既支持简单的timer,也支持灵活强大的corn表达式。本文提供了一种把Quartz.Net嵌入项目中的实现。你可以看到系统其它项目用于作业调度的项目解耦,这样做可以实现一次封装,多系统使用

1、环境

  • 操作系统:Windows 10
  • IDE:Visual Studio 2019-16.8.3
  • .Net Core版本:.NET 5.0

2、创建项目,添加引用

创建一个空解决方案Theo.QuartzDemo,然后开始添加新项目。

2.1、创建名为Theo.Business的类库项目

创建.Net Core类库项目Theo.Business,代表系统的业务项目。添加Microsoft.Extensions.Logging引用。

添加IProviderDemo接口,定义两个方法分别提供发邮件和发短信服务。你可能觉得发邮件和发短信服务不应该在同一个provider中,但是作为示例,我偷了个懒。

IProviderDemo.cs:

using System;

namespace Theo.Business
{
public interface IProviderDemo
{ /// <summary>
/// 模拟提供发邮件服务
/// </summary>
/// <param name="param">参数</param>
void AutoSendMail(string param); /// <summary>
/// 模拟提供发短信服务
/// </summary>
/// <param name="param">参数</param>
void AutoSendSMS(string param);
}
}

添加IProviderDemo的实现:ProviderDemo

using Microsoft.Extensions.Logging;
using System; namespace Theo.Business
{
///<summary>
/// 模拟业务逻辑
///</summary>
public class ProviderDemo : IProviderDemo
{
private readonly ILogger<ProviderDemo> _logger; /// <summary>
/// .ctor
/// </summary>
/// <param name="logger"></param>
public ProviderDemo(ILogger<ProviderDemo> logger)
{
_logger = logger;
} /// <summary>
/// 模拟提供发邮件服务
/// </summary>
/// <param name="param">参数</param>
public void AutoSendMail(string param)
{
_logger.LogError($"[{DateTimeOffset.Now:HH:mm:ss.fff}]\t{nameof(AutoSendMail)}\tparam:{param}");
} /// <summary>
/// 模拟提供发短信服务
/// </summary>
/// <param name="param">参数</param>
public void AutoSendSMS(string param)
{
_logger.LogInformation($"[{DateTimeOffset.Now:HH:mm:ss.fff}]\t{nameof(AutoSendSMS)}\tparam:{param}");
}
}
}

2.2、添加名为Theo.TaskSchedulerWorker Service(辅助角色服务)项目

创建Theo.TaskScheduler项目,用于作业调度



添加引用

Install-Package Microsoft.Extensions.Hosting.Systemd
Install-Package Microsoft.Extensions.Hosting.WindowsServices
Install-Package Quartz.AspNetCore
Install-Package Microsoft.Extensions.Logging.Log4Net.AspNetCore
Install-Package System.Text.Json
  • Microsoft.Extensions.Hosting.Systemd:用以支持Linux守护进程,部署在Linux平台时需要此组件
  • Microsoft.Extensions.Hosting.WindowsServices:用以支持Windows服务,部署在Windows平台时需要此组件
  • Quartz.AspNetCore:.Net Core平台的Quartz组件
  • Microsoft.Extensions.Logging.Log4Net.AspNetCore:Log4Net日志组件,当然也可以是Nlog/Serilog等其他任意组件
  • 添加对Theo.Business项目的引用

3、封装Quartz.Net

Theo.TaskScheduler项目中封装Quartz.Net。封装的主要目的是为了更方便的配置作业调度计划。实现过程中的工作量主要在读取配置并交给Quartz.Net组件,以告诉组件如何根据配置去调用指定方法。

新建Models文件夹,文件夹下新建JobModelJobGroupModelScheduleModel三个类。

* JobModel:计划任务job模型

* JobGroupModel:计划任务job组模型

* ScheduleModel:作业调度模型

新建IOCHelper文件,注入业务代码(IProviderDemo),并暴露IServiceProvider

新建Quartz文件夹,文件夹下新建:

  • QuartzJob:实现Quartz.IJobExecute方法,来调用配置的作业调度。因为Quartz.AspNetCore目前对IOC的支持有限,这里实现一个protected void Init(ILogger logger, IServiceProvider serviceProvider)方法,以便在Execute方法中记录日志以及通过IServiceProvider发现服务。Execute方法的逻辑:

    • IJobExecutionContext读取配置
    • 通过IServiceProvider发现服务
    • Invoke调用
  • BlockedJob:继承QuartzJob,类添加[DisallowConcurrentExecution]属性,指定为阻塞模式job。例如某个job调用频率是每分钟1次,本次调用时发现上次调用还未执行结束,将放弃本次调用且不会抛出任何异常
  • ConcurrentJob:继承QuartzJob,可并发执行job,不管上次执行是否结束,到时间就重新调用执行

    新建QuartzWorker类,继承BackgroundService并重载StopAsyncExecuteAsync方法。完成读取作业调度配置文件并交给Quartz.Net组件的工作。通过appsetting.json文件Quartz:Config配置找到作业调度配置文件路径,读取配置文件。配置文件支持xmljson两种格式二选一。根据Quartz:Watch配置确定是否监控配置文件变化。如果Quartz:Watch配置为true,在服务运行过程中可以通过修改作业调度配置文件来修改作业调度计划。

    新建Configs文件夹,文件夹下新建SchedulerConfig.json或者SchedulerConfig.xml配置文件,用以配置作业调度计划。SchedulerConfig.json配置示例如下,发邮件服务调用频率为从5秒开始每10秒钟一次,发短信服务调用频率为从0秒开始每10秒钟一次
{
"GroupList": [
{
"Name": "JobGroup",
"JobList": [
{
"Disabled": false,
"Desc": "发邮件服务",
"Name": "SendMail",
"Cron": "5/10 * * * * ? *",
"DllName": "Theo.Business",
"ClassName": "Theo.Business.IProviderDemo",
"MethodName": "AutoSendMail"
},
{
"Disabled": false,
"Desc": "发短信服务",
"Name": "SendSMS",
"Cron": "0/10 * * * * ? *",
"DllName": "Theo.Business",
"ClassName": "Theo.Business.IProviderDemo",
"MethodName": "AutoSendSMS"
}
]
}
]
}

appsetting.json添加配置,指定作业调度配置文件:

...
"Quartz": {
"Config": "Configs/SchedulerConfig.json",
"Watch": true
}
...

4、配置Program并测试运行

Program.cs文件:

using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
using System;
using Theo.TaskScheduler.Quartz; namespace Theo.TaskScheduler
{
public class Program
{
public static void Main(string[] args)
{
CreateHostBuilder(args).Build().Run();
} public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.UseWindowsService() //支持Windows服务, 其他平台自动忽略
.UseSystemd() //支持Linux守护进程,其他平台自动忽略
//使用log4net日志组件,并指定配置文件
.ConfigureLogging(conf => conf.AddLog4Net("Configs/log4net.config"))
.ConfigureAppConfiguration((hostContext, config) =>
{
config.SetBasePath(AppContext.BaseDirectory);
var env = hostContext.HostingEnvironment;
if (env.IsDevelopment())
{
config.AddJsonFile("appsettings.Development.json");
}
else
{
config.AddJsonFile("appsettings.json");
}
})
.ConfigureServices((hostContext, services) =>
{
IOCHelper.InjectDependencies(services);
services.AddHostedService<QuartzWorker>();
});
}
}

启动调试之前还需要为Log4Net日志组件添加配置文件。根据Program.cs中指定的配置文件Configs/log4net.config,我们在项目Configs文件夹下添加文件log4net.config。具体的配置规则您可以去 log4net官网 查找,篇幅所限,本文不做说明。

终于可以F5启动调试,结果如下图所示。

我们的作业调度配置:SendMail从5秒开始每10秒钟一次SendSMS从0秒开始每10秒钟一次。从下图可以看出,作业调度运行情况和我们预期的一致。



在运行中,我们把SendMail作业改为从1秒开始每10秒钟一次,把SendSMS作业禁用,然后新增一个名为SendSMS-1的作业,调用频率为从0秒开始每3秒钟一次。从下图可以看出(``开始),程序检测到了配置文件的变化,并启用的新的作业调度计划。

5、部署服务

部署到Linux平台的方法,您可以参阅官方文档:

若要部署到Windows平台,您可以发布Theo.TaskScheduler项目,然后通过执行shell命令或者把命令包装成.bat文件并执行的方式安装为Windows服务并启动。参考命令如下:

@echo.服务启动......
@echo off
@sc create TheoTaskScheduler binPath= "publish\Theo.TaskScheduler.exe"
@sc description TheoTaskScheduler "TheoDemo-任务调度服务"
@net start TheoTaskScheduler
@echo off
@echo.启动完毕!
@pause

6、总结

本文介绍了Quartz.Net组件在DotNetCore平台封装使用的详细步骤。

Quartz.AspNetCore封装到了单独DotNetCore项目中,与其他业务代码解耦。并实现了作业调度计划的配置化,和服务运行中实时监控配置文件功能。可以在运行中动态禁用/启用作业,添加新的作业,修改现有作业的调度计划(cron表达式)。

上文有提到示例中实现了可以指定是否允许并行调用作业的功能(BlockedJob/ConcurrentJob),但因篇幅所限并未贴出测试结果。

展望:

1、在大型项目中,调度服务可能以集群或者分布式架构的形式运行。这就要考虑把作业调度计划配置到数据库或者redis等缓存中。服务启动时候从数据库或者缓存中加载作业调度计划,并订阅修改作业调度计划事件,以便在调度计划发生改变时及时更新。

2、本例仅实现了调度服务运行中实时禁用/启用作业,添加新的作业,修改现有作业的调度计划的功能,以满足基本使用需求。如果您有兴趣/需求,可以自行探索如何实时修改是否允许并行调用作业等功能。

示例代码已上传至资源库,仅需5积分,感谢您的支持:Theo.QuartzDemo源码

Quartz.Net 组件的封装使用Quartz.AspNetCore的更多相关文章

  1. ASP.NET MVC(C#)和Quartz.Net组件

    ASP.NET MVC(C#)和Quartz.Net组件 在之前的文章<推荐一个简单.轻量.功能非常强大的C#/ASP.NET定时任务执行管理器组件–FluentScheduler>和&l ...

  2. 控制台基于Quartz.Net组件实现定时任务调度(一)

    前言: 你曾经需要应用执行一个任务吗?比如现在有一个需求,需要每天在零点定时执行一些操作,那应该怎样操作呢? 这个时候,如果你和你的团队是用.NET编程的话,可以考虑使用Quartz.NET调度器.允 ...

  3. quartz.net插件类库封装(含源码)

    1.前言 目录: 1.quartz.net任务调度:源码及使用文档 2.quartz.net插件类库封装 最近项目需要做一写任务作业调度的工作,最终选择了quartz.net这个插件,它提供了巨大的灵 ...

  4. 基于MVC 的Quartz.Net组件实现的定时执行任务调度

    新建mvc项目之后,首先引用Quartz组件.工具-->NuGet包管理器-->管理解决方案的 NuGet包管理器 组件安装完成. Quartz.Net一个最简单任务至少包括三部分实现:j ...

  5. MVC 使用Quartz.Net组件实现定时计划任务

    最近,项目中需要执行一个计划任务,组长就让我了解一下Quartz.net 这个组件,挺简单的一个组件,实现起来特别的方便,灵活,值得推荐给大家一起学习一下这个小工具.以前我有的时候是使用定时器Time ...

  6. c# Quartz.net的简单封装

    分享一个以前封装的Quartz.net类. 新建一个QuartzClass类库项目.nuget控制台输入 image.png 添加Quartz.net的引用. 我们新建一个JobBase.cs文件,里 ...

  7. Window服务基于Quartz.Net组件实现定时任务调度(二)

    前言: 在上一章中,我们通过利用控制台实现定时任务调度,已经大致了解了如何基于Quartz.Net组件实现任务,至少包括三部分:job(作业),trigger(触发器),scheduler(调度器). ...

  8. Quartz简单实现定时任务管理(SSM+Quartz)

    首先你得有一个用Maven搭好的SSM框架,数据库用的Mysql,这里只有关于Quartz的部分.其实有大神总结的很好了,但做完后总有些地方不一样,所以写这篇作为笔记.这里先把大神的写的分享给大家:h ...

  9. 支付sdk —— 该组件为封装了 微信,支付宝,银联支付

    [精品]  支付组件 简要说明该组件为封装了 微信,支付宝,银联支付, 一键快速集成,几行代码即可集成 微信,支付宝,银联支付. ## 示例: # 测试账号:1.银联支付:提供测试使用卡号.手机号信息 ...

随机推荐

  1. String 类的常用方法都有那些?

    1.indexOf():返回指定字符的索引. 2.charAt():返回指定索引处的字符. 3.replace():字符串替换. 4.trim():去除字符串两端空白. 5.split():分割字符串 ...

  2. Java基础经典案例

    案例列表 01减肥计划switch版本 02减肥计划if版本 03逢七跳过 04不死神兔 05百钱白鸡 06数组元素求和 07判断两个数组是否相同 08查找元素在数组中的索引 09数组元素反转 10评 ...

  3. String -- 从源码剖析String类

    几乎所有的 Java 面试都是以 String 开始的,String 源码属于所有源码中最基础.最简单的一个,对 String 源码的理解也反应了你的 Java 基础功底. String 是如何实现的 ...

  4. 【对线面试官】Java NIO

    服务端: public class NoBlockServer { public static void main(String[] args) throws IOException { // 1.获 ...

  5. OpenOCD安装与使用(JTAG调试)

    本文介绍openocd开源软件的安装以及搭配JTAG对Xilinx u500VC707devkit的调试 PC OS: Ubuntu20.04 LTS Target ARCH: riscv64 JTA ...

  6. leetcode230. 二叉搜索树中第K小的元素

    题目链接: https://leetcode-cn.com/problems/kth-smallest-element-in-a-bst/ 题目: 给定一个二叉搜索树,编写一个函数 kthSmalle ...

  7. DNS基础概要

    dns服务系统由客户端和服务器组成,提供域名到ip地址的解析,或者提供ip地址到域名的逆向解析. 1.DNS域名空间 每个dns域名由分级的label构成,如www.sina.com.cn,由www. ...

  8. linux多路径(multipath)

    https://www.itread01.com/articles/1475909423.html

  9. 24V转3.3V稳压芯片,高效率同步降压DC-DC变换器3A输出电流

    PW2312是一个高频,同步,整流,降压,开关模式转换器与内部功率MOSFET.它提供了一个非常紧凑的解决方案,以实现1.5A的峰值输出电流在广泛的输入电源范围内,具有良好的负载和线路调节. PW23 ...

  10. requests模块的基本使用

    requests模块的基本使用 基于网络请求的模块. 环境的安装:pip install requests 作用:模拟浏览器发起请求 分析requests的编码流程: 1.指定url 2.发起了请求 ...