在.NET Core 中使用Quartz.NET-程序员宅基地

技术标签: python  区块链  java  linux  docker  

Quartz.NET是功能齐全的开源作业调度系统,可用于最小的应用程序到大型企业系统。

Quartz.NET具有三个主要概念:

•job:运行的后台任务•trigger:控制后台任务运行的触发器。•scheduler:协调job和trigger

ASP.NET Core通过托管服务对运行“后台任务”具有良好的支持,托管服务在ASP.NET Core应用程序启动时启动,并在应用程序生存期内在后台运行,Quartz.NET版本3.2.0通过Quartz.Extensions.Hosting包引入了对该模式的直接支持,Quartz.Extensions.Hosting可以与ASP.NET Core应用程序一起使用,也可以与基于“通用主机”的工作程序服务一起使用。

虽然.NET Core可以创建“定时”后台服务(例如,每10分钟运行一次任务),但Quartz.NET提供了更为强大的解决方案, 通过使用Cron表达式,您可以确保任务在特定时间(例如,凌晨2:30)运行,或仅在特定的几天运行,或这些时间的任意组合。Quartz.NET还允许您以集群方式运行应用程序的多个实例,以便在任何时候都只能运行一个实例。

安装Quartz.NET

Quartz.NET是一个.NET Standard 2.0 NuGet软件包,所以大部分项目都是支持的,你可以运行安装命令,dotnet add package Quartz.Extensions.Hosting,或者在NNuget可视化安装,如果查看该项目的.csproj,应该是下边这样:

<Project Sdk="Microsoft.NET.Sdk.Worker">


  <PropertyGroup>
    <TargetFramework>net5.0</TargetFramework>
    <UserSecretsId>dotnet-QuartzWorkerService-9D4BFFBE-BE06-4490-AE8B-8AF1466778FD</UserSecretsId>
  </PropertyGroup>


  <ItemGroup>
    <PackageReference Include="Microsoft.Extensions.Hosting" Version="5.0.0" />
    <PackageReference Include="Quartz.Extensions.Hosting" Version="3.2.3" />
  </ItemGroup>
</Project>

安装完成以后,这个包会自动安装 Quartz.NET包,接下来,我们需要在我们的应用程序中注册Quartz服务和Quartz 。

添加Quartz.NET hosted service

修改Program.cs,注册服务

public class Program
{
    public static void Main(string[] args)
    {
        CreateHostBuilder(args).Build().Run();
    }


    public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureServices((hostContext, services) =>
            {
                // Add the required Quartz.NET services
                services.AddQuartz(q =>  
                {
                    // Use a Scoped container to create jobs. I'll touch on this later
                    q.UseMicrosoftDependencyInjectionScopedJobFactory();
                });


                // Add the Quartz.NET hosted service


                services.AddQuartzHostedService(
                    q => q.WaitForJobsToComplete = true);


                // other config
            });
}

UseMicrosoftDependencyInjectionScopedJobFactory(),这个地方告诉Quartz.NET注册一个IJobFactory,然后从DI容器中获取Job,这样也可以使用 Scoped 类型的服务。

WaitForJobsToComplete():当程序关闭时,此设置可确保Quartz.NET在退出之前等待Job正常结束。

如果现在运行您的应用程序,您将看到Quartz服务启动,并将有很多日志输出到控制台:

info: Quartz.Core.SchedulerSignalerImpl[0]
      Initialized Scheduler Signaller of type: Quartz.Core.SchedulerSignalerImpl
info: Quartz.Core.QuartzScheduler[0]
      Quartz Scheduler v.3.2.3.0 created.
info: Quartz.Core.QuartzScheduler[0]
      JobFactory set to: Quartz.Simpl.MicrosoftDependencyInjectionJobFactory
info: Quartz.Simpl.RAMJobStore[0]
      RAMJobStore initialized.
info: Quartz.Core.QuartzScheduler[0]
      Scheduler meta-data: Quartz Scheduler (v3.2.3.0) 'QuartzScheduler' with instanceId 'NON_CLUSTERED'
  Scheduler class: 'Quartz.Core.QuartzScheduler' - running locally.
  NOT STARTED.
  Currently in standby mode.
  Number of jobs executed: 0
  Using thread pool 'Quartz.Simpl.DefaultThreadPool' - with 10 threads.
  Using job-store 'Quartz.Simpl.RAMJobStore' - which does not support persistence. and is not clustered.


info: Quartz.Impl.StdSchedulerFactory[0]
      Quartz scheduler 'QuartzScheduler' initialized
info: Quartz.Impl.StdSchedulerFactory[0]
      Quartz scheduler version: 3.2.3.0
info: Quartz.Core.QuartzScheduler[0]
      Scheduler QuartzScheduler_$_NON_CLUSTERED started.
info: Microsoft.Hosting.Lifetime[0]
      Application started. Press Ctrl+C to shut down.
...

现在,您已经将Quartz作为托管服务运行在您的应用程序中,但是现在还没有添加需要运行的Job。

创建一个IJob

这个地方我创建一个简单的服务,并且我可以从构造函数中获取服务。

using Microsoft.Extensions.Logging;
using Quartz;
using System.Threading.Tasks;


[DisallowConcurrentExecution]
public class HelloWorldJob : IJob
{
    private readonly ILogger<HelloWorldJob> _logger;
    public HelloWorldJob(ILogger<HelloWorldJob> logger)
    {
        _logger = logger;
    }


    public Task Execute(IJobExecutionContext context)
    {
        _logger.LogInformation("Hello world!");
        return Task.CompletedTask;
    }
}

我还用[DisallowConcurrentExecution]特性,防止Quartz.NET尝试同时运行同一个作业。

设置Job

这个地方通常使用Cron表达式,来设置job的执行时间。

public static IHostBuilder CreateHostBuilder(string[] args) =>
    Host.CreateDefaultBuilder(args)
        .ConfigureServices((hostContext, services) =>
        {
            services.AddQuartz(q =>  
            {
                q.UseMicrosoftDependencyInjectionScopedJobFactory();


                // Create a "key" for the job
                var jobKey = new JobKey("HelloWorldJob");


                // Register the job with the DI container
                q.AddJob<HelloWorldJob>(opts => opts.WithIdentity(jobKey));


                // Create a trigger for the job
                q.AddTrigger(opts => opts
                    .ForJob(jobKey) // link to the HelloWorldJob
                    .WithIdentity("HelloWorldJob-trigger") // give the trigger a unique name
                    .WithCronSchedule("0/5 * * * * ?")); // run every 5 seconds


            });
            services.AddQuartzHostedService(q => q.WaitForJobsToComplete = true);
            // ...
        });

现在运行应用程序,您将看到和以前相同的启动消息,然后每隔5秒钟就会看到HelloWorldJob写入控制台的信息:

将配置提取到appsettings.json

一般情况,我们都不会把cron表达式写死在代码中,一般是设置在appsettings.json中

{
  "Quartz": {
    "HelloWorldJob": "0/5 * * * * ?"
  }
}

为了更简单的注册服务,这个地方我简单做了一个封装,这样也更灵活。

public static class ServiceCollectionQuartzConfiguratorExtensions
{
    public static void AddJobAndTrigger<T>(
        this IServiceCollectionQuartzConfigurator quartz,
        IConfiguration config)
        where T : IJob
    {
        // Use the name of the IJob as the appsettings.json key
        string jobName = typeof(T).Name;


        // Try and load the schedule from configuration
        var configKey = $"Quartz:{jobName}";
        var cronSchedule = config[configKey];


        // Some minor validation
        if (string.IsNullOrEmpty(cronSchedule))
        {
            throw new Exception($"No Quartz.NET Cron schedule found for job in configuration at {configKey}");
        }


        // register the job as before
        var jobKey = new JobKey(jobName);
        quartz.AddJob<T>(opts => opts.WithIdentity(jobKey));


        quartz.AddTrigger(opts => opts
            .ForJob(jobKey)
            .WithIdentity(jobName + "-trigger")
            .WithCronSchedule(cronSchedule)); // use the schedule from configuration
    }
}

然后修改Program.cs,然后使用扩展方法:

public class Program
{
    public static void Main(string[] args) => CreateHostBuilder(args).Build().Run();


    public static IHostBuilder CreateHostBuilder(string[] args) =>
        Host.CreateDefaultBuilder(args)
            .ConfigureServices((hostContext, services) =>
            {
                services.AddQuartz(q =>
                {
                    q.UseMicrosoftDependencyInjectionScopedJobFactory();


                    // Register the job, loading the schedule from configuration
                    q.AddJobAndTrigger<HelloWorldJob>(hostContext.Configuration);
                });


                services.AddQuartzHostedService(q => q.WaitForJobsToComplete = true);
            });
}

再次运行该应用程序将提供相同的输出:Job每5秒输入一次信息。

原文作者: andrewlock 

原文链接: https://andrewlock.net/using-quartz-net-with-asp-net-core-and-worker-services/[1]

最后

欢迎扫码关注我们的公众号 【全球技术精选】,专注国外优秀博客的翻译和开源项目分享,也可以添加QQ群 897216102

References

[1] https://andrewlock.net/using-quartz-net-with-asp-net-core-and-worker-services/: "https://andrewlock.net/using-quartz-net-with-asp-net-core-and-worker-services/"

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/sD7O95O/article/details/113152657

智能推荐

不兼容android5.1.1,Android5.1.1启动问题-程序员宅基地

文章浏览阅读903次。本帖最后由 xujin071 于 2016-12-12 11:27 编辑新买回的firefly-rk3288开发板,烧入最新的Android5.1.1编译的固件,一直卡在Android字符界面,进不了桌面,各位版主帮忙分析一下启动log信息。[ 2.338220] ======== PULL WL_REG_ON LOW! ========[ 2.338228] [WLAN_RFKILL..._tk_btusb( 0): btchr_open: device not probed

python截图黑屏_对Python获取屏幕截图的4种方法详解-程序员宅基地

文章浏览阅读2.6k次,点赞2次,收藏2次。Python获取电脑截图有多种方式,具体如下:PIL中的ImageGrab模块windows APIPyQtpyautoguiPIL中的ImageGrab模块import timeimport numpy as npfrom PIL import ImageGrabimg = ImageGrab.grab(bbox=(100, 161, 1141, 610))img = np.array(img...._win32gui截屏黑屏

易语言代码转换python_易语言通过文本解析的方式把C代码转换成易代码-程序员宅基地

文章浏览阅读1.1k次。常量数据表.版本 2.常量 c, "", , '常量值是一段C代码C代码转易代码.版本 2.支持库 commobj.支持库 iext2.程序集 窗口程序集_启动窗口.程序集变量 k, 快速文本对象.程序集变量 k2, 快速文本对象.子程序 __启动窗口_创建完毕.局部变量 z, 字符格式z.字体大小 = 8z.字体名称 = “微软雅黑”d1.置默认字符格式 (z)d2.置默认字符格式 (z)d1...._易语言转python

python绘制蟒蛇_如何采用Python语言绘制一条彩色的蟒蛇-程序员宅基地

文章浏览阅读311次。2.编写drawSnake:该函数数用来实现画蛇的具体操作,主要涉及到到画图的方法;对于该函数,我们首先得确定画蛇的基本策略,该函数中用到了很多的turtle包中的子函数,具体可参见该包的帮助文档,这里贴出代码:def drawSnake(rad,angle,len,neckrad):mycolor=["black","red","red","blue","yellow"]yocolor=["ye..._编写程序,绘制任意颜色任意长度的蟒蛇

oracle和sql数据库,SQL与ORACLE异同-程序员宅基地

文章浏览阅读398次。001、SQL与ORACLE的内存分配ORACLE的内存分配大部分是由INIT.ORA来决定的,一个数据库实例可以有N种分配方案,不同的应用(OLTP、OLAP)它的配置是有侧重的。SQL概括起来说,只有两种内存分配方式:动态内存分配与静态内存分配,动态内存分配充许SQL自己调整需要的内存,静态内存分配限制了SQL对内存的使用。002、SQL与ORACLE的物理结构总得讲,它们的物理结构很相似,S..._oracle数据库与sql数据库的区别

sundancest201驱动_FreeNAS 支持的网卡列表-程序员宅基地

文章浏览阅读3.4k次。### 3.2.Ethernet InterfacesThe ae(4) driver supports Attansic/Atheros L2 PCIe FastEthernet controllers, and is known to support the following hardware:- ASUS EeePC 701- ASUS EeePC 900Other hardware m..._freenas truenas 支持的网卡

随便推点

java 地图 聚类_【TOAN HOANG 专题(42)】聚类地图-程序员宅基地

文章浏览阅读254次。本文搬运自国外Tableau大神原创文章,Tableau交流问答群为国内唯一独家授权组织Toan Hoang:知名Tableau大神,数据可视化自由职业者和Tableau Magic的创始人,萨尔萨舞教练,钢琴演奏者,技术爱好者和程序员。Toan Hoang另本文由Tableau交流问答群Tableau爱好者—Lynn对原文进行翻译,若有问题,欢迎讨论~前言一般的,我们在做地图分布分析的时候,只是..._java 站点 聚类

idea解决javaweb项目servlet 404 找不到对应servlet_idea怎么看servlet版本-程序员宅基地

文章浏览阅读6.4k次,点赞13次,收藏37次。哎。今天是制造bug的一天。idea下的javaweb项目,写的servlet一直404各种碰壁,最终解决了。下面先说一下问题的原因:servlet2.5是要手写web.xml <servlet> <servlet-name>Servlet</servlet-name> <servl..._idea怎么看servlet版本

表单下拉框动态获取数据_k-form-design下拉选动态数据-程序员宅基地

文章浏览阅读4.2k次。1.在进行表单操作时,下拉框选项需要动态读取数据库数据,使用ajax异步获取数据。&lt;form class="layui-form layui-form-pane" method="post" id="websiteform"&gt; &lt;div class="layui-form-item"&gt; &lt;label class="layui-for_k-form-design下拉选动态数据

Linux配置终端代理_linux终端代理-程序员宅基地

文章浏览阅读1.4w次,点赞7次,收藏24次。Linux配置终端代理_linux终端代理

MySQL学习笔记整理(下部)-程序员宅基地

文章浏览阅读5.8k次。以下是我整理了关于MySQL数据库的常用命令和使用场景,一些基本的概念,比如索引、约束和事务,以及常见的问题和解决方案,欢迎大家浏览并留言,若有错误的地方请大家指正。

java菜鸟程序员2012年度总结——分享、收获与感恩并存-程序员宅基地

文章浏览阅读91次。前言:又是一年总结时啊。本来总结打算前几天就该写的。但由于一直在忙最后的期末考试,今天终于考完了。现在终于有时间来对这一年进行总一下了。刚开始的时候想了半天不知道该用什么题目好。想了想,今年的博客一直围绕着“java程序员由笨鸟到菜鸟”的专题来写的。现在想想。经过一年的洗礼,差不多由一个没有思想,没有想法的笨鸟变成会敲两行代码的菜鸟了。所以有了现在的“java菜鸟程序员2012总结”题..._程序员感恩总结