作業調度系統quartz.net

来源:https://www.cnblogs.com/net-yuan/archive/2018/12/13/quartz.html
-Advertisement-
Play Games

任務調度在我們日常開發過程中非常常見,比如:每天晚上0點自動執行某某操作;每周三晚上2點執行某某操作;......當然,我們處理這類問題的方法也有很多,比如:sql的自動任務;windows上創建任務計劃;寫windows服務等等。如果系統比較複雜,相互調用比較頻繁,任務非常多,幾百上千條甚至上萬條 ...


        任務調度在我們日常開發過程中非常常見,比如:每天晚上0點自動執行某某操作;每周三晚上2點執行某某操作;......當然,我們處理這類問題的方法也有很多,比如:sql的自動任務;windows上創建任務計劃;寫windows服務等等。如果系統比較複雜,相互調用比較頻繁,任務非常多,幾百上千條甚至上萬條,那麼本身對任務的管理就是比較昂貴的代價;如何提高任務的高可用?任務的測試是否便捷等等問題就會出現。上述的方案是否還能從容應對?

        這時我們就迫切地需要一個作業調度系統來處理這些場景。

        Quartz.NET是一個強大、開源、輕量的作業調度框架,是 OpenSymphony 的 Quartz API 的.NET移植,它有很多特征,如:資料庫支持,集群,插件,支持cron-like表達式等等。官方網址:https://www.quartz-scheduler.net;GitHub地址:

https://github.com/quartznet/quartznet,各種用法可以參考示常式序。

        但如果想方便的知道某個作業執行情況,需要暫停,啟動等操作行為,這時候就需要個Job管理的界面。CrystalQuartz可以實現遠程管理。

        多數系統都會涉及到“後臺服務”的開發,一般是為了調度一些自動執行的任務或從隊列中消費一些消息,開發 windows service 有一點不爽的是:調試麻煩,當然你還需要知道 windows service 相關的一些開發知識(也不難),TopShelf框架,可以你讓 console application 封裝為 windows service,這樣你就非常方便的開發和調試 windows service。TopShelf框架的官網為Url:http://topshelf-project.com

        用TopShelf和quartz.net編寫任務,CrystalQuartz管理任務。本文就是搭建一個簡易的任務調度方案,啟動任務調度、添加Job、移除Job、遠程管理等。

        1、建立一個名為TaskScheduling的Asp.net MVC項目

       

        2、新建一個類RemoteServerExample,主要代碼:

 1 //ILog log = LogManager.GetLogger(typeof(RemoteServerExample));
 2 
 3             NameValueCollection properties = new NameValueCollection();
 4             properties["quartz.scheduler.instanceName"] = "RemoteAutoTaskServer";
 5 
 6             properties["quartz.threadPool.type"] = "Quartz.Simpl.SimpleThreadPool, Quartz";
 7             properties["quartz.threadPool.threadCount"] = "5";
 8             properties["quartz.threadPool.threadPriority"] = "Normal";
 9 
10             // 設置伺服器
11             properties["quartz.scheduler.exporter.type"] = "Quartz.Simpl.RemotingSchedulerExporter, Quartz";
12             properties["quartz.scheduler.exporter.port"] = "555";
13             properties["quartz.scheduler.exporter.bindName"] = "QuartzScheduler";
14             properties["quartz.scheduler.exporter.channelType"] = "tcp";
15 
16             #region 2.3.0版本可用
17 
18             //properties["quartz.scheduler.exporter.channelName"] = "httpQuartz";
19             // 拒絕遠程
20             //properties["quartz.scheduler.exporter.rejectRemoteRequests"] = "true";
21 
22             #endregion 2.3.0版本可用
23 
24             #region 持久化所用
25 
26             //存儲類型
27             //properties["quartz.jobStore.type"] = "Quartz.Impl.AdoJobStore.JobStoreTX, Quartz";
28             ////表明首碼
29             //properties["quartz.jobStore.tablePrefix"] = "QRTZ_";
30             ////驅動類型
31             //properties["quartz.jobStore.driverDelegateType"] = "Quartz.Impl.AdoJobStore.SqlServerDelegate, Quartz";
32             ////數據源名稱
33             //properties["quartz.jobStore.dataSource"] = "myDS";
34             ////連接字元串
35             //properties["quartz.dataSource.myDS.connectionString"] = @"Data Source=(local);Initial Catalog=quartz;User ID=sa;Password=Ayy123";
36             ////sqlserver版本
37             //properties["quartz.dataSource.myDS.provider"] = "SqlServer-20";
38 
39             #endregion 持久化所用
40 
41             ISchedulerFactory sf = new StdSchedulerFactory(properties);
42             IScheduler sched = sf.GetScheduler();
43             sched.Start();
44 
45             SchedulerMetaData metaData = sched.GetMetaData();
46 
47             IJobDetail job = JobBuilder.Create<SimpleJob>()
48                 .WithIdentity("remotelyAddedJob", "default")
49                 .Build();
50 
51             JobDataMap map = job.JobDataMap;
52             map.Put("msg", "信息");
53 
54             ITrigger trigger = TriggerBuilder.Create()
55                 .WithIdentity("remotelyAddedTrigger", "default")
56                 .ForJob(job.Key)
57                 .WithCronSchedule("/5 * * ? * *")
58                 .Build();
59 
60             string name = sched.SchedulerName;
61             sched.ScheduleJob(job, trigger);
View Code

3、新建一個類RemoteClientExample,主要代碼:

 1 ILog log = LogManager.GetLogger(typeof(RemoteClientExample));
 2 
 3             NameValueCollection properties = new NameValueCollection();
 4             properties["quartz.scheduler.instanceName"] = "RemoteClient";
 5 
 6             // 設置線程池
 7             properties["quartz.threadPool.type"] = "Quartz.Simpl.SimpleThreadPool, Quartz";
 8             properties["quartz.threadPool.threadCount"] = "5";
 9             properties["quartz.threadPool.threadPriority"] = "Normal";
10 
11             // 設置遠程連接
12             //properties["quartz.scheduler.proxy"] = "true";
13             //properties["quartz.scheduler.proxy.address"] = "tcp://127.0.0.1:556/QuartzScheduler";
14 
15             ISchedulerFactory sf = new StdSchedulerFactory(properties);
16             IScheduler scheduler = sf.GetScheduler("RemoteAutoTaskServer");
17 
18             IJobDetail job = JobBuilder.Create<SimpleJob>()
19                 .WithIdentity("remotelyAddedJob1", "default")
20                 .Build();
21 
22             JobDataMap map = job.JobDataMap;
23             map.Put("msg", "信息");
24 
25             ITrigger trigger = TriggerBuilder.Create()
26                 .WithIdentity("remotelyAddedTrigger1", "default")
27                 .ForJob(job.Key)
28                 .WithCronSchedule("/5 * * ? * *")
29                 .Build();
30 
31             string name = scheduler.SchedulerName;
32             scheduler.ScheduleJob(job, trigger);
33 
34             log.Info("向伺服器添加計劃任務");
View Code

4、SimpleJob代碼:

 1 [PersistJobDataAfterExecution]
 2     [DisallowConcurrentExecution]
 3     public class SimpleJob : IJob
 4     {
 5         public const string Message = "msg";
 6 
 7         public virtual void Execute(IJobExecutionContext context)
 8         {
 9             try
10             {
11                 JobKey jobKey = context.JobDetail.Key;
12                 string message = context.JobDetail.JobDataMap.GetString(Message);
13 
14                 string strLog = String.Format("{0} 執行時間 {1}", jobKey, DateTime.Now.ToString());
15             }
16             catch (Exception ex)
17             {
18             }
19         }
20     }
View Code

5、程式運行界面:

點擊啟動調度器,啟動伺服器監聽且預設有一個名為remotelyAddedJob的job,並且允許遠程管理;點擊添加任務,會加入一個名為remotelyAddedJob1的job;刪除任務會刪除所有job。

如果我們已經啟動了調度器,有一個任務,比如,我們用TopShelf和quartz.net寫了一個windows服務,在本地運行,如果要加入到遠程管理中,怎麼辦呢?

可以按照上述RemoteClientExample代碼,配置properties,調用的時候通過反射的方式獲取,然後加入到調度器中,進行遠程管理。

quartz.net還有持久化、集群等等特性,提高任務的高可用。

分享一個corn表達式生成器:https://files.cnblogs.com/files/net-yuan/CronExpBuilder.zip

大神張善友 的quartz.net系列:https://www.cnblogs.com/shanyou/category/102991.html

GitHub:https://github.com/anangyang/TaskScheduling

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 入門示例 異常處理:try/except 對於索引查找的操作,在索引越界搜索的時候會報錯。例如: 所報的錯誤是IndexError。如果將索引查找放在一個函數里: 那麼調用函數的時候,如果裡面的索引越界了,異常將彙報到函數調用者。 可以使用try/except來捕獲異常。作為入門示例,下麵是簡單版的 ...
  • 一 內置函數 1. revserd 翻轉,返回的是迭代器 2.slice 切片 3.formate 4. type() 返回類型 ord() 輸入字元找字元編碼的位置 chr() 輸入位置找出對應的字元 ascii()判斷給出的信息是否是ascii 二. 遞歸 函數自己調用自己,遞歸的入口(參數) ...
  • 上一章講了pod的管理,今天再分享一個pod的訪問方式 1.Pod的HostIP模式 Pod的HostIP模式,可以通過宿主機訪問pod內的服務,創建yaml文件如下 直接create 我們去192.1268.8.202節點去查看一下2000埠 訪問一下192.168.8.202:2000 埠正 ...
  • 需求場景:在查詢頁面,填寫查詢條件,查詢條件包括上傳的圖片,根據圖片的特征查詢,這就需要在提交的時候,使用POST提交,因為GET提交無法提交圖片數據,提交查詢條件之後,在新的視窗展示查詢結果。(當然查詢結果頁面可能不支持F5刷新頁面) 表單HTML代碼示意(註意method="post" targ ...
  • 網路 TCP:與打電話類似,通知服務到位 UDP:與發簡訊類似,消息發出即可 IP和埠號是網路兩大重要成員 埠號(Port)分為知名埠號[0-1024,不開放)和動態埠號[1024,10000多,開放可用) 三次握手,四次揮手: unity網端簡單案例: 分為:綜合管理部分、客戶端和伺服器 ...
  • 在我們開發Winform界面的時候,往往需要綁定數據字典操作,也就是綁定一些下拉列表或者一些列表顯示等,以便我們方便選擇數據操作,常見的字典綁定操作就是對下拉列表的處理,本篇隨筆是基於DevExpress界面的一些處理操作,原理也適用於常規Winform界面或者DotNetBar控制項界面處理。另外對... ...
  • 在同一個功能變數名稱下有很多子系統 如:a.giant.com b.giant.com c.giant.com等 但是這些系統都是giant.com這個子域。 這樣的情況就可以在不引用其它框架的情況下,直接基於Cookie實現同域單點登錄SSO 註:用ID4,OAuth,其它SSO框架也同樣可以實現。本文不 ...
  • 什麼是遞歸函數? 任何一個方法既可以調用其他方法又可以調用自己,而當這個方法調用自己時,我們就叫它遞歸函數或者遞歸方法! 說白了,就是調用自己。 通常遞歸有兩個特點: 1.遞歸方法一直會調用自己直到某些條件滿足,也就是說一定要有出口; 2.遞歸方法會有一些參數,而它會把這些新的參數值傳遞給自己;(自 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...