使用 Topshelf 组件一步一步创建 Windows 服务 (2) 使用Quartz.net 调度
上一篇说了如何使用 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文件 都需要在项目中设置,右键-->属性-->复制到输出目录-->始终复制
这里提供一个在线生成 cron表达式地址: https://cron.qqe2.com
服务注册文件 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