使用 Topshelf 組件一步一步創建 Windows 服務 (2) 使用Quartz.net 調度

来源:https://www.cnblogs.com/peterzhang123/archive/2019/11/29/11908742.html
-Advertisement-
Play Games

上一篇說瞭如何使用 Topshelf 組件快速創建Windows服務,接下來介紹如何使用 Quartz.net 關於Quartz.net的好處,網上搜索都是一大把一大把的,我就不再多介紹。 先介紹需要用到的插件: Quartz版本我用的 2.6.2的, 沒有用3.0以上的,因為你用了就會知道,會列印 ...


上一篇說瞭如何使用 Topshelf 組件快速創建Windows服務,接下來介紹如何使用 Quartz.net

關於Quartz.net的好處,網上搜索都是一大把一大把的,我就不再多介紹。

先介紹需要用到的插件:

Quartz版本我用的 2.6.2的, 沒有用3.0以上的,因為你用了就會知道,會列印出一大堆坑爹的日誌文件,

我是沒有找到如何屏蔽的辦法,如果你們誰有,歡迎分享出來,我也學習一下,哈哈。

整個項目結構如下:

AppConfigHelper 文件需要改動一下,增加如下屬性
 1         /// <summary>
 2         /// 程式標識
 3         /// </summary>
 4         [ConfigurationProperty("AppKey", IsRequired = true)]
 5         public string AppKey
 6         {
 7             get { return base["AppKey"].ToString(); }
 8             internal set { base["AppKey"] = value; }
 9         }
10 
11         /// <summary>
12         /// 程式集信息
13         /// </summary>
14         [ConfigurationProperty("TypeInfo", IsRequired = true)]
15         public string TypeInfo
16         {
17             get { return base["TypeInfo"].ToString(); }
18             internal set { base["TypeInfo"] = value; }
19         }

AppConfig文件也做稍微改動

 1 <?xml version="1.0" encoding="utf-8" ?>
 2 <configuration>
 3   <!--該節點一定要放在最上邊-->
 4   <configSections>
 5     <section name="AppConfigHelper" type="Quartz.WinService.AppConfigHelper,Quartz.WinService"/>
 6   </configSections>
 7 
 8   <!--TopSelf服務配置文件 -->
 9   <AppConfigHelper
10     ServiceName="ProcessPrintLogService"
11     Desc="日誌列印服務"
12     AppKey="ProcessPrintLogService"
13     TypeInfo="ProcessService.ProcessPrintLogService,ProcessService"
14   />
15 
16   <startup>
17     <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5.2" />
18   </startup>
19 </configuration>
ProcessPrintLogService 就是Windows服務要執行的邏輯程式文件,可以執行任何你想要的功能
ProcessService.ProcessPrintLogService,ProcessService 是 命名空間.類名,類名  的格式,用於後邊反射程式集用

假如你要執行其他業務邏輯程式,只需要更換這裡的配置就行,
ProcessPrintLogService 業務邏輯內容如下:這就是我們要執行的業務邏輯,定時列印一段日誌內容,可以創建一個類庫,裡邊專門存放你要執行的業務邏輯
 1 namespace ProcessService
 2 {
 3     /// <summary>
 4     /// 日誌列印服務
 5     /// </summary>
 6     public class ProcessPrintLogService
 7     {
 8         private Logger log = LogManager.GetCurrentClassLogger();
 9         /// <summary>
10         /// 服務入口
11         /// </summary>
12         public void DoWork()
13         {
14             //log.Info("******************排行榜服務開始執行******************");
15             try
16             {
17                 PrintLogMethod();
18             }
19             catch (Exception ex)
20             {
21                 log.Error(string.Format("排行榜服務異常,原因:{0}", ex));
22             }
23             finally
24             {
25                 //log.Info("******************排行榜服務結束執行******************");
26             }
27         }
28 
29 
30         private void PrintLogMethod()
31         {
32             log.Trace(string.Format("我是日誌:{0}號", Thread.CurrentThread.ManagedThreadId));
33         }
34     }
35 }

然後需要新增加兩個文件:quartz.config  和  quartz_jobs.xml

quartz.config文件內容如下:

# You can configure your scheduler in either <quartz> configuration section
# or in quartz properties file
# Configuration section has precedence

quartz.scheduler.instanceName = ServiceQuartzScheduler

# configure thread pool info
quartz.threadPool.type = Quartz.Simpl.SimpleThreadPool, Quartz
quartz.threadPool.threadCount = 10
quartz.threadPool.threadPriority = Normal

# job initialization plugin handles our xml reading, without it defaults are used
quartz.plugin.xml.type = Quartz.Plugin.Xml.XMLSchedulingDataProcessorPlugin, Quartz
quartz.plugin.xml.fileNames = ~/quartz_jobs.xml

# 3.0以上用以下配置
# quartz.plugin.xml.type = Quartz.Plugin.Xml.XMLSchedulingDataProcessorPlugin, Quartz.Plugins
# quartz.plugin.xml.fileNames = ~/quartz_jobs.xml

# export this server to remoting context
# quartz.scheduler.exporter.type = Quartz.Simpl.RemotingSchedulerExporter, Quartz
# quartz.scheduler.exporter.port = 555
# quartz.scheduler.exporter.bindName = QuartzScheduler
# quartz.scheduler.exporter.channelType = tcp
# quartz.scheduler.exporter.channelName = httpQuartz
quartz.scheduler.instanceName = ServiceQuartzScheduler  是調度的實例名稱,可以隨意自定義命名
其他的都是固定的,不需要修改
quartz_jobs.xml 文件內容如下:
<?xml version="1.0" encoding="UTF-8"?>

<job-scheduling-data xmlns="http://quartznet.sourceforge.net/JobSchedulingData" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="2.0">

  <processing-directives>
    <overwrite-existing-data>true</overwrite-existing-data>
  </processing-directives>
  <schedule>
    <!--調度配置-->
    <job>
      <name>ProcessPrintLogService</name>
      <group>ProcessPrintLogServiceGroup</group>
      <description>日誌列印服務</description>
      <job-type>Quartz.WinService.QuartzWork,Quartz.WinService</job-type>
      <durable>true</durable>
      <recover>false</recover>
    </job>
    <trigger>
      <cron>
        <name>ProcessPrintLogServiceTrigger</name>
        <group>ProcessPrintLogServiceTriggerGroup</group>
        <job-name>ProcessPrintLogService</job-name>
        <job-group>ProcessPrintLogServiceGroup</job-group>
        <misfire-instruction>SmartPolicy</misfire-instruction>
        <cron-expression>0/3 * * * * ? </cron-expression>
      </cron>
    </trigger>
  </schedule>
</job-scheduling-data>

這個xml配置文件很重要! 需要重點說下

首先 job節點 和 trigger節點 都可以定義多個,也就是一個服務可以跑多個不同的業務邏輯程式

先說 job節點

  • name(必填) 任務名稱,多個job的name不能相同,這裡一般使用業務邏輯程式的名稱就行了
  • group(選填) 任務所屬分組,用於標識任務所屬分組,一般用業務邏輯程式的名稱+Group尾碼   如:<group>sampleGroup</group>
  • description(選填) 任務描述,用於描述任務具體內容,如:<description>列印日誌服務</description>
  • job-type(必填) 任務類型,任務的具體類型及所屬程式集,格式:實現了IJob介面的包含完整命名空間的類名,程式集名稱,如:<job-type>Quartz.Server.SampleJob, Quartz.Server</job-type>
  • durable(選填) 具體作用不知,官方示例中預設為true,如:<durable>true</durable>
  • recover(選填) 具體作用不知,官方示例中預設為false,如:<recover>false</recover>

這裡的 job-type 節點調用的任務類型需要說下,這裡設置的就是上邊項目結構中的 QuartzWork 類,具體內容如下:

namespace Quartz.WinService
{
    public class QuartzWork : IJob
    {
        private Logger log = LogManager.GetCurrentClassLogger();
        //ConcurrentDictionary是線程安全的字典集
        private readonly ConcurrentDictionary<string, Lazy<Delegate>> _dynamicCache = new ConcurrentDictionary<string, Lazy<Delegate>>();

        //記錄當前工作介面是否已經工作
        private static readonly Dictionary<string, bool> WorkingNow = new Dictionary<string, bool>();

        /// <summary>
        /// 任務調度執行入口
        /// 實現IJob的Execute方法,在Execute方法里編寫要處理的業務邏輯,系統就會按照Quartz的配置,定時處理
        /// 當Job的trigger觸發的時候, Execute(..) 方法就會在scheduler的工作線程中執行
        /// </summary>
        /// <param name="context"></param>
        public void Execute(IJobExecutionContext context)
        {
            try
            {
                Task.Factory.StartNew(() =>
                {
                    var service = AppConfigHelper.Initity();
                    WorkNow(service);
                });
            }
            catch (Exception ex)
            {
                log.Fatal($"執行Quartz調度異常,信息:{ex.Message}");
            }
            //return Task.FromResult(true);  //返回一個bool類型的Task, Quartz 3.0版本以上需要用到
        }

        private void WorkNow(AppConfigHelper service)
        {
            string key = service.AppKey;  //key值
            lock (this)
            {
                if (!WorkingNow.ContainsKey(key))
                {
                    WorkingNow.Add(key, false);
                }
                //如果執行則跳出
                if (WorkingNow[key])
                {
                    log.Trace($"服務key:{key} 正在運行,此次服務忽略");
                    return;
                }
                //並且設置為執行狀態
                WorkingNow[key] = true;
            }
            try
            {
                var type = Type.GetType(service.TypeInfo);  //這裡通過App.config文件設置
                if (type != null)
                {
                    //創建指定類型的實例,相當於通過反射new了一個對象實例
                    var provider = Activator.CreateInstance(type);
                    Dynamic(provider, "DoWork", key);
                }
                else
                {
                    log.Error($"任務:{key} 實例化失敗");
                }
            }
            catch (Exception ex)
            {
                log.Fatal($"任務:{key} 實例化異常:{ex.Message}");
            }
            finally
            {
                WorkingNow[key] = false;
            }
        }

        //Delegate.CreateDelegate 官方定義:用來動態創建指定類型的委托,該委托可以對指定的類實例調用的指定的方法。
        //簡單來說:就是可以調用指定類裡邊指定的方法,前提是,使用時需要實例化該類
        //GetOrAdd函數會根據指定key判斷是否存在對應內容,存在則返回
        //DynamicInvoke 動態調用委托方法
        //obj參數就是指定類的實例化對象,methodName指定類中的方法名
        private void Dynamic(object obj, string methodName, string key)
        {
            var dmc = _dynamicCache.GetOrAdd(key, t => new Lazy<Delegate>(() => Delegate.CreateDelegate(typeof(Action), obj, methodName)));
            dmc.Value.DynamicInvoke();   //動態調用委托方法
        }

    }
}

接下來說 trigger  節點

trigger 任務觸發器,用於定義使用何種方式出發任務(job),同一個job可以定義多個trigger ,多個trigger 各自獨立的執行調度,

每個trigger 中必須且只能定義一種觸發器類型(calendar-interval、simple、cron)

說白些就是,假如你要一個服務分別在 上午 8:00~18:00   和  凌晨 00:00 ~ 6:00  這兩個時間段執行任務,那麼你可以設置兩個 trigger 觸發器,

分別設置為這兩個時間段即可實現你要的結果,怎麼樣,很牛X吧

  • name(必填) 觸發器名稱,一般以 業務邏輯類+Trigger結尾, 如果需要設置多個 trigger節點,該名稱不能相同
  • group(選填) 觸發器組  一般以 業務邏輯類+TriggerGroup結尾,多個 trigger節點,該名稱可以相同
  • job-name(必填) 要調度的任務名稱,該job-name必須和對應job節點中的name名稱完全相同
  • job-group(選填) 調度任務(job)所屬分組,該值必須和job節點中的group名稱完全相同
  • misfire-instruction 不知道幹啥用,這麼寫就行  <misfire-instruction>SmartPolicy</misfire-instruction>
  • cron-expression(必填) cron表達式,如:<cron-expression>0/10 * * * * ?</cron-expression>每10秒執行一次

需要註意的是修改了quartz_jobs.xml文件後,quartz服務預設不會重新載入該文件,若要讓修改後的文件生效需要重啟下服務才行。

另外,quartz.config文件 和 quartz_jobs.xml文件 都需要在項目中設置,右鍵-->屬性-->複製到輸出目錄-->始終複製

 

服務註冊文件 RegistService 增加了自動重啟功能,完整內容如下:

namespace Quartz.WinService
{
    public class RegistService
    {
        /// <summary>
        /// 註冊入口
        /// </summary>
        /// <param name="config">配置文件</param>
        /// <param name="isreg">是否註冊</param>
        public static void Regist(AppConfigHelper config, bool isreg = false)
        {
            //這裡也可以使用HostFactory.Run()代替HostFactory.New()
            var host = HostFactory.New(x =>
            {
                x.Service<QuartzHost>(s =>
                {
                    //通過 new QuartzHost() 構建一個服務實例 
                    s.ConstructUsing(name => new QuartzHost());
                    //當服務啟動後執行什麼
                    s.WhenStarted(tc => tc.Start());
                    //當服務停止後執行什麼
                    s.WhenStopped(tc => tc.Stop());
                    //當服務暫停後執行什麼
                    s.WhenPaused(w => w.Stop());
                    //當服務繼續後執行什麼
                    s.WhenContinued(w => w.Start());
                });

                if (!isreg) return; //false表示不註冊

                //服務用本地系統賬號來運行
                x.RunAsLocalSystem();

                //啟用自動重啟服務
                x.EnableServiceRecovery(v =>
                {
                    v.RestartService(2);  //2分鐘後重啟
                });

                //服務的描述信息
                x.SetDescription(config.Description);
                //服務的顯示名稱
                x.SetDisplayName(config.ServiceName);
                //服務的名稱(最好不要包含空格或者有空格屬性的字元)Windows 服務名稱不能重覆。
                x.SetServiceName(config.ServiceName);
            }).Run();   //啟動服務  如果使用HostFactory.Run()則不需要該方法
        }
    }
}

服務註冊中調用的 QuartzHost 類內容如下:

namespace Quartz.WinService
{
    public class QuartzHost
    {
        private Logger log = LogManager.GetCurrentClassLogger();
        private readonly IScheduler scheduler;
        public QuartzHost()
        {
            //初始化調度服務
            //scheduler = StdSchedulerFactory.GetDefaultScheduler().Result;  //3.0以上寫法
            scheduler = StdSchedulerFactory.GetDefaultScheduler();
        }

        /// <summary>
        /// 調度開始
        /// </summary>
        public void Start()
        {
            try
            {
                scheduler.Start();
                log.Info("Quartz調度服務開始工作");
            }
            catch (Exception ex)
            {
                log.Fatal(string.Format("Quartz調度服務開始異常!錯誤信息:{0}", ex));
                throw;
            }
        }

        /// <summary>
        /// 調度停止
        /// </summary>
        public void Stop()
        {
            try
            {
                if (scheduler != null)
                {
                    scheduler.Shutdown(true);
                }
                log.Info("Quartz調度服務結束工作");
            }
            catch (Exception ex)
            {
                log.Fatal(string.Format("Quartz調度服務停止異常!錯誤信息:{0}", ex));
                throw;
            }
        }
    }
}

項目文件地址:https://gitee.com/gitee_zhang/Quartz.WinService.git


參考文檔:

https://blog.csdn.net/clb929/article/details/90341485

https://blog.csdn.net/weixin_33948416/article/details/92989386

https://www.cnblogs.com/lzrabbit/archive/2012/04/14/2446942.html


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

-Advertisement-
Play Games
更多相關文章
  • 後臺的配置 1.創建後臺管理員 [root@localhost study_django]# python manage.py createsuperuser [root@localhost study_django]# ./manage.py createsuperuser 2.啟動服務,運行項目 ...
  • 正在陸續開源自己的一些項目 此為c++實現高併發的游戲盒子,平臺問題需要遷移重構,所以有一些遺留問題,客戶端異常斷開沒有處理,會導致伺服器崩潰,還有基於快寫代碼編程平臺實現的小程式切換,線程讀寫緩存沒有加鎖可能有衝突,導致閃退,不切換裡面的小程式可以正常使用,剩下的等完善自己的手機端編程IDE,會基 ...
  • dict = {}for i in range(1, 6): if i not in dict: dict[i] = [] for j in range(101, 106): dict[i].append(j)print(dict) ...
  • I/O模型之BIO 基本介紹 Java BIO 就是傳統的 Java IO 編程,其相關的類和介面再 java.io 包下 BIO(blocking I/O):同步阻塞,伺服器實現模式為一個連接一個線程,即客戶端有連接請求時伺服器端就需要啟動一個線程進行處理,但是如果新啟動的這一個線程不做任何事情就 ...
  • 對於非純字元串組成的列表,需要使用map(str, 列表)轉換,純字元串組成的列表則不需要轉換 ...
  • 1.python之logger日誌通用配置文件 2.放入項目即可直接使用 ...
  • Intellj IDEA快捷鍵入門 之 Ctrl+Space(空格) 時間 :2019/11/28 系統 :Win10系統 背景 : 步驟 : 1.win 設置 時間和語言 語言 添加語言 2.這裡選最基本的就好,不想選亂七八糟的,點擊"安裝" 3.安裝完成後(重啟完成後),這裡就會有兩種語言,把E ...
  • 正則匹配並且可以捕獲到()這個裡面的子表達式的值,linux的grep命令沒辦法捕獲子表達式的值,只能獲取到整條正則匹配的內容 上面的正則中驗證了.*是貪婪 .*?是非貪婪 ,下麵匹配的字元串切片第一條是整條數據,後面的每一個對應正則括弧里捕獲的內容 tao@tao-PC:/var/www/html ...
一周排行
    -Advertisement-
    Play Games
  • C#TMS系統代碼-基礎頁面BaseCity學習 本人純新手,剛進公司跟領導報道,我說我是java全棧,他問我會不會C#,我說大學學過,他說這個TMS系統就給你來管了。外包已經把代碼給我了,這幾天先把增刪改查的代碼背一下,說不定後面就要趕鴨子上架了 Service頁面 //using => impo ...
  • 委托與事件 委托 委托的定義 委托是C#中的一種類型,用於存儲對方法的引用。它允許將方法作為參數傳遞給其他方法,實現回調、事件處理和動態調用等功能。通俗來講,就是委托包含方法的記憶體地址,方法匹配與委托相同的簽名,因此通過使用正確的參數類型來調用方法。 委托的特性 引用方法:委托允許存儲對方法的引用, ...
  • 前言 這幾天閑來沒事看看ABP vNext的文檔和源碼,關於關於依賴註入(屬性註入)這塊兒產生了興趣。 我們都知道。Volo.ABP 依賴註入容器使用了第三方組件Autofac實現的。有三種註入方式,構造函數註入和方法註入和屬性註入。 ABP的屬性註入原則參考如下: 這時候我就開始疑惑了,因為我知道 ...
  • C#TMS系統代碼-業務頁面ShippingNotice學習 學一個業務頁面,ok,領導開完會就被裁掉了,很突然啊,他收拾東西的時候我還以為他要旅游提前請假了,還在尋思為什麼回家連自己買的幾箱飲料都要叫跑腿帶走,怕被偷嗎?還好我在他開會之前拿了兩瓶芬達 感覺感覺前面的BaseCity差不太多,這邊的 ...
  • 概述:在C#中,通過`Expression`類、`AndAlso`和`OrElse`方法可組合兩個`Expression<Func<T, bool>>`,實現多條件動態查詢。通過創建表達式樹,可輕鬆構建複雜的查詢條件。 在C#中,可以使用AndAlso和OrElse方法組合兩個Expression< ...
  • 閑來無聊在我的Biwen.QuickApi中實現一下極簡的事件匯流排,其實代碼還是蠻簡單的,對於初學者可能有些幫助 就貼出來,有什麼不足的地方也歡迎板磚交流~ 首先定義一個事件約定的空介面 public interface IEvent{} 然後定義事件訂閱者介面 public interface I ...
  • 1. 案例 成某三甲醫預約系統, 該項目在2024年初進行上線測試,在正常運行了兩天後,業務系統報錯:The connection pool has been exhausted, either raise MaxPoolSize (currently 800) or Timeout (curren ...
  • 背景 我們有些工具在 Web 版中已經有了很好的實踐,而在 WPF 中重新開發也是一種費時費力的操作,那麼直接集成則是最省事省力的方法了。 思路解釋 為什麼要使用 WPF?莫問為什麼,老 C# 開發的堅持,另外因為 Windows 上已經裝了 Webview2/edge 整體打包比 electron ...
  • EDP是一套集組織架構,許可權框架【功能許可權,操作許可權,數據訪問許可權,WebApi許可權】,自動化日誌,動態Interface,WebApi管理等基礎功能於一體的,基於.net的企業應用開發框架。通過友好的編碼方式實現數據行、列許可權的管控。 ...
  • .Net8.0 Blazor Hybird 桌面端 (WPF/Winform) 實測可以完整運行在 win7sp1/win10/win11. 如果用其他工具打包,還可以運行在mac/linux下, 傳送門BlazorHybrid 發佈為無依賴包方式 安裝 WebView2Runtime 1.57 M ...