日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

.NET6+Quartz實現定時任務的示例詳解

瀏覽:392日期:2022-06-11 09:29:41
目錄
  • 什么是定時任務
  • 什么是Quartz
  • 涉及知識點
  • Quartz安裝
  • 創建一個簡單的定時器任務
    • 1. 創建工作單元Job
    • 2. 創建時間軸Scheduler
    • 3. 創建觸發規則Trigger
    • 4. 創建任務描述
    • 5. 建立三者聯系
    • 6. 簡單示例測試
  • 傳遞參數
    • 任務特性
      • 監聽器
        • 1. 創建監聽器
        • 2. 添加監聽
      • 日志管理
        • 完整示例

          在實際工作中,經常會有一些需要定時操作的業務,如:定時發郵件,定時統計信息等內容,那么如何實現才能使得我們的項目整齊劃一呢?本文通過一些簡單的小例子,簡述在.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接口,如下所示:

          using Quartz;using System.Diagnostics;namespace DemoQuartz.QuartzA.Job{    /// <summary>    /// 測試任務,實現IJob接口    /// </summary>    public class TestJob : IJob    {public TestJob(){    Console.WriteLine("執行構造函數");//表示每一次計劃執行,都是一次新的實例}public Task Execute(IJobExecutionContext context){    return Task.Run(() =>     { Console.WriteLine($"******************************"); Console.WriteLine($"測試信息{DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss.fff")}"); Console.WriteLine($"******************************"); Console.WriteLine();     });}    }}

          2. 創建時間軸Scheduler

          時間軸也是任務執行的載體,可以通過StdSchedulerFactory進行獲取,如下所示:

          //創建計劃單元(時間軸,載體) StdSchedulerFactory schedulerFactory = new StdSchedulerFactory(); var scheduler = await schedulerFactory.GetScheduler(); await scheduler.Start();

          3. 創建觸發規則Trigger

          觸發規則就是那些時間點執行任務,可通過TriggerBuilder進行構建,如下所示:

          //Trigger時間觸發機制var trigger = TriggerBuilder.Create()    .WithIdentity("TestTrigger","TestGroup")    //.StartNow() //立即執行    .WithSimpleSchedule(w=>w.WithIntervalInSeconds(5).WithRepeatCount(5))//.RepeatForever()//無限循環    //.WithCronSchedule("5/10 * * * * ?") //通過Cron表達式定制時間觸發規則, 示例表示從5開始,每隔10秒一次    .Build();

          4. 創建任務描述

          任務描述定義了具體的任務名稱,分組等內容。可通過JobBuilder進行構建,如下所示:

          //Job詳細描述 var jobDetail = JobBuilder.Create<TestJob>()     .WithDescription("這是一個測試Job")     .WithIdentity("TestJob", "TestGroup")     .Build();

          5. 建立三者聯系

          通過載體,將規則和工作單元串聯起來,如下所示:

           //把時間和任務通過載體關聯起來 await scheduler.ScheduleJob(jobDetail, trigger);

          6. 簡單示例測試

          通過運行程序,示例結果如下所示:

          傳遞參數

          在Quartz框架下,如果需要給執行的Job傳遞參數,可以通過兩種方式:

          jobDetail.JobDataMap,工作描述時通過JobDataMap傳遞參數。

          trigger.JobDataMap, 時間觸發時通過JobDataMap傳遞參數。

          在Job工作單元中,可以通過Context中對應的JobDataMap獲取參數。

          傳遞參數,如下所示:

          //傳遞參數jobDetail.JobDataMap.Add("name", "Alan");jobDetail.JobDataMap.Add("age", 20);jobDetail.JobDataMap.Add("sex", true);//trigger同樣可以傳遞參數trigger.JobDataMap.Add("like1", "meimei");trigger.JobDataMap.Add("like2", "football");trigger.JobDataMap.Add("like3", "sing");

          獲取參數,如下所示:

          //獲取參數var name = context.JobDetail.JobDataMap.GetString("name");var age = context.JobDetail.JobDataMap.GetInt("age");var sex = context.JobDetail.JobDataMap.GetBoolean("sex") ? "男" : "女";var like1 = context.Trigger.JobDataMap.GetString("like1");var like2 = context.Trigger.JobDataMap.GetString("like2");var like3 = context.Trigger.JobDataMap.GetString("like3");//context.MergedJobDataMap.GetString("aa");//注意如果使用MergedJobDataMap,JobDetail和Trigger中用到相同的Key,則后面設置的會覆蓋前面設置的。

          注意:如果使用MergedJobDataMap,JobDetail和Trigger中用到相同的Key,則后面設置的會覆蓋前面設置的。

          任務特性

          假如我們的定時任務,執行一次需要耗時比較久,而且后一次執行需要等待前一次完成,并且需要前一次執行的結果作為參考,那么就需要設置任務的任性。因為默認情況下,工作單元在每一次運行都是一個新的實例,相互之間獨立運行,互不干擾。所以如果需要存在一定的關聯,就要設置任務的特性,主要有兩個,如下所示:

          • [PersistJobDataAfterExecution]//在執行完成后,保留JobDataMap數據
          • [DisallowConcurrentExecution]//不允許并發執行,即必須等待上次完成后才能執行下一次

          以上兩個特性,只需要標記在任務對應的類上即可。標記上后,只需要往對應的JobDataMap中添加值即可。

          監聽器

          在Quartz框架下,有三種監聽器,分別是:時間軸監聽器ISchedulerListener,觸發規則監聽器ITriggerListener,任務監聽器IJobListener。要實現對應監聽器,實現對應接口即可。實現監聽器步驟:

          1. 創建監聽器

          根據不同的需要,可以創建不同的監聽器,如下所示:

          時間軸監聽器SchedulerListener

          public class TestSchedulerListener : ISchedulerListener{    public Task JobAdded(IJobDetail jobDetail, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Job is added.");});    }    public Task JobDeleted(JobKey jobKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Job is deleted.");});    }    public Task JobInterrupted(JobKey jobKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Job is Interrupted.");});    }    public Task JobPaused(JobKey jobKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Job is paused.");});    }    public Task JobResumed(JobKey jobKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Job is resumed.");});    }    public Task JobScheduled(ITrigger trigger, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Job is scheduled.");});    }    public Task JobsPaused(string jobGroup, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Jobs is paused.");});    }    public Task JobsResumed(string jobGroup, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Jobs is resumed.");});    }    public Task JobUnscheduled(TriggerKey triggerKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test Jobs is un schedulered.");});    }    public Task SchedulerError(string msg, SchedulerException cause, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduler is error.");});    }    public Task SchedulerInStandbyMode(CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduler is standby mode.");});    }    public Task SchedulerShutdown(CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduler is shut down.");});    }    public Task SchedulerShuttingdown(CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduler is shutting down.");});    }    public Task SchedulerStarted(CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduleer is started.");});    }    public Task SchedulerStarting(CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduler is starting.");});    }    public Task SchedulingDataCleared(CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test scheduling is cleared.");});    }    public Task TriggerFinalized(ITrigger trigger, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test trigger is finalized.");});    }    public Task TriggerPaused(TriggerKey triggerKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test trigger is paused.");});    }    public Task TriggerResumed(TriggerKey triggerKey, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test trigger is resumed.");});    }    public Task TriggersPaused(string? triggerGroup, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test triggers is paused.");});    }    public Task TriggersResumed(string? triggerGroup, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test triggers is resumed.");});    }}

          觸發規則監聽器TriggerListener

          /// <summary>/// 觸發器監聽/// </summary>public class TestTriggerListener : ITriggerListener{    public string Name => "TestTriggerListener";    public Task TriggerComplete(ITrigger trigger, IJobExecutionContext context, SchedulerInstruction triggerInstructionCode, CancellationToken cancellationToken = default)    {//任務完成return Task.Run(() => {    Console.WriteLine("Test trigger is complete.");});    }    public Task TriggerFired(ITrigger trigger, IJobExecutionContext context, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test trigger is fired.");});    }    public Task TriggerMisfired(ITrigger trigger, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test trigger is misfired.");});    }    public Task<bool> VetoJobExecution(ITrigger trigger, IJobExecutionContext context, CancellationToken cancellationToken = default)    {return Task.Run(() => {    Console.WriteLine("Test trigger is veto.");    return false;//是否終止});    }}

          JobListener任務監聽器

          /// <summary>/// TestJob監聽器/// </summary>public class TestJobListener : IJobListener{    public string Name => "TestJobListener";    public Task JobExecutionVetoed(IJobExecutionContext context, CancellationToken cancellationToken = default)    {//任務被終止時return Task.Run(() => {    Console.WriteLine("Test Job is vetoed.");});    }    public Task JobToBeExecuted(IJobExecutionContext context, CancellationToken cancellationToken = default)    {//任務被執行時return Task.Run(() => {    Console.WriteLine("Test Job is to be executed.");});    }    public Task JobWasExecuted(IJobExecutionContext context, JobExecutionException? jobException, CancellationToken cancellationToken = default)    {//任務已經執行return Task.Run(() => {    Console.WriteLine("Test Job was executed.");});    }}

          2. 添加監聽

          在時間軸上的監聽管理器中進行添加,如下所示:

          //增加監聽 scheduler.ListenerManager.AddJobListener(new TestJobListener()); scheduler.ListenerManager.AddTriggerListener(new TestTriggerListener()); scheduler.ListenerManager.AddSchedulerListener(new TestSchedulerListener());

          日志管理

          在Quartz框架中,創建之前會進行日志創建檢測,所以如果需要獲取框架中的日志信息,可以進行創建實現ILogProvider,如下所示:

          public class TestLogProvider : ILogProvider{    public Logger GetLogger(string name)    {return (level, func, exception, parameters) =>{    if (level >= Quartz.Logging.LogLevel.Info && func != null)    {Console.WriteLine("[" + DateTime.Now.ToLongTimeString() + "] [" + level + "] " + func(), parameters);    }    return true;};    }    public IDisposable OpenMappedContext(string key, object value, bool destructure = false)    {throw new NotImplementedException();    }    public IDisposable OpenNestedContext(string message)    {throw new NotImplementedException();    }}

          然后在當前的Scheduler中,添加日志即可,如下所示:

           //日志 LogProvider.SetCurrentLogProvider(new TestLogProvider());

          完整示例

          在添加了監聽器,日志,參數傳遞,任務特性后,完整的目錄結構,如下所示:

          示例截圖

          到此這篇關于.NET6+Quartz實現定時任務的示例詳解的文章就介紹到這了,更多相關.NET6 Quartz定時任務內容請搜索以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持!

          標簽: ASP.NET
          日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
          国产精品久久久免费| 日本成人在线网站| 国产成人免费| 啪啪国产精品| 亚洲深深色噜噜狠狠爱网站 | 国内一区二区三区| 久久男人av| 久久精品动漫| 一区二区三区网站| 老鸭窝一区二区久久精品| 国产精品毛片久久久| 黑人精品一区| 伊人久久亚洲美女图片| 国产视频欧美| 国产精品毛片久久久| 国产精品三级| 亚洲不卡av不卡一区二区| 影音先锋久久| 亚洲人亚洲人色久| 日韩欧美精品一区| 蜜桃av一区二区在线观看| 91亚洲国产成人久久精品| 热三久草你在线| 亚洲三级av| 久久av一区二区三区| 男人的天堂久久精品| 久久精品一区二区国产| 久久久久99| 日韩国产在线观看| 亚洲va在线| 欧美激情精品| 日韩欧美中文字幕电影 | 亚洲二区视频| 日韩中文字幕| 欧美日韩国产在线一区| 亚洲女同中文字幕| 欧美国产日本| 国产真实久久| 日韩精品一级| 美女精品在线| 欧美freesex黑人又粗又大| 中文不卡在线| 麻豆精品视频在线| 在线免费观看亚洲| 亚洲成人二区| 激情久久五月| 国产一区二区三区亚洲综合| 亚洲精品黄色| 午夜精品久久久久久久久久蜜桃| 久久精品72免费观看| 91精品一区二区三区综合| 国产精品视频一区二区三区综合| 91精品久久久久久久久久不卡| 在线日韩视频| 久久99影视| 日本aⅴ免费视频一区二区三区| 精品三区视频| 日韩av午夜在线观看| 91精品在线观看国产| 久久中文字幕导航| jizzjizz中国精品麻豆| 久久中文字幕一区二区三区| 亚洲三级毛片| 石原莉奈在线亚洲三区| 欧美成人精品午夜一区二区| 91精品丝袜国产高跟在线| 国产精品女主播一区二区三区| 国产精品**亚洲精品| 国产美女精品| 欧美日韩国产免费观看视频| 精品亚洲a∨一区二区三区18| 影音先锋久久精品| 欧美羞羞视频| 国产精品成人一区二区不卡| 国产精品一区毛片| 午夜欧美精品| 日韩综合精品| 国产精品xxx| 久久精品国产在热久久| 视频一区欧美日韩| 热久久免费视频| 综合激情视频| 999国产精品999久久久久久| 久久精品动漫| 91精品国产调教在线观看| 国产精品久久久久9999高清| 国产精品久久久一区二区| 国产精选久久| 国产日韩欧美一区在线| 99成人在线| 粉嫩av一区二区三区四区五区 | 中文精品电影| 精品91久久久久| 欧美午夜不卡| 爽爽淫人综合网网站| 亚洲天堂资源| 久久中文字幕av| 一区福利视频| 日韩在线黄色| 日本在线成人| 欧美精品导航| 国产资源在线观看入口av| 国产999精品在线观看| 国产精品久久久久77777丨| 国产精品久久久久久久免费观看| 亚洲va中文在线播放免费| 桃色一区二区| 亚洲一区二区成人| 色综合视频一区二区三区日韩 | 高清日韩中文字幕| 国产精品天堂蜜av在线播放| 日韩国产激情| 九一精品国产| 一区二区亚洲精品| 亚洲精品第一| 精品免费av| 久久精品72免费观看| 亚洲精品伊人| 美腿丝袜在线亚洲一区| 日韩欧美精品一区二区综合视频| 国产欧美亚洲一区| 欧美永久精品| 亚洲三级欧美| 日韩在线欧美| 国产精品婷婷| 国产精品一级在线观看| 日本免费新一区视频| 牛牛精品成人免费视频| 国产一在线精品一区在线观看| 欧美日韩视频免费观看| 91精品精品| 四虎精品永久免费| 欧美日韩国产高清电影| 亚洲第一区色| 91精品丝袜国产高跟在线| 久久久精品日韩| 亚洲成人av观看| 午夜一级久久| 国产亚洲一区二区三区啪| 欧美日韩亚洲一区| 亚洲午夜天堂| 日韩高清电影一区| 国产情侣一区| 欧美福利一区| 欧美黄色精品| 欧美日韩一区二区三区不卡视频| 日韩免费在线| 一二三区精品| 91精品二区| 国产乱码精品一区二区三区四区| 久久精品99国产精品日本| 欧美日韩夜夜| 精品99久久| 欧美精品一区二区三区精品| 91久久在线| 亚洲精品第一| 精品国产日韩欧美精品国产欧美日韩一区二区三区 | 夜夜精品视频| 日本韩国欧美超级黄在线观看| 亚洲高清激情| 久久久夜夜夜| 色老板在线视频一区二区| 好看的av在线不卡观看| 亚洲精品高潮| 欧美日本不卡| 国产精品videossex| 日韩国产欧美| 蜜桃视频免费观看一区| 欧美在线精品一区| 亚洲激精日韩激精欧美精品| 欧美午夜不卡| 都市激情国产精品| 亚洲激情偷拍| 国产剧情一区| 亚洲成av人片一区二区密柚| 爽好多水快深点欧美视频| 国产极品嫩模在线观看91精品| 精品中文在线| 久久三级视频| 欧美精品影院| 精品国产美女a久久9999| 欧美午夜精彩| 久久国内精品视频| 久久免费大视频| 久久狠狠亚洲综合| 激情欧美一区| 国产极品嫩模在线观看91精品| 中文在线日韩| 精品一区91| 免费看日韩精品| 国产h片在线观看| 视频精品一区二区| 中文在线а√在线8| 日韩欧美中文字幕在线视频| 日韩毛片视频| 亚洲字幕久久| 999国产精品999久久久久久| 国产私拍福利精品视频二区| 婷婷精品进入| 精品久久91|