在系统运维中常常需要定期去跑一些计划任务,比如扫描服务器监控其性能、检查SQL Server作业是否正常、监控MQ队列是否存在堵塞现象等。如果使用Windows计划任务调度,一来管理起来就比较松散,二来如需更改计划任务的配置就必须登录到服务器上进行修改,造成很大的不便。因此笔者在实际工作中自行开发计划任务调度服务来处理这些任务,将调度周期、任务配置等经常需要修改的配置信息保存到数据库中,并开发一个前台界面进行维护和管理。

一、基本结构

计划任务调度服务使用插件的方式处理各类不同的计划任务,插件必须继承自服务框架提供的MonitorTask抽象类,并在数据库中注册任务名、调度周期等信息,这样就可以由该服务调度执行。为方便起见,要求任务名、插件的dll文件名、插件对象的类型名必须一致,各插件都放在同一个命名空间MonitorTask下。

在工作中,各类监控任务主要有3种配置:

1.调度周期,以xml形式保存在数据库中。

2.监控项,以键值对的方式保存在数据库中。

3.基本不会修改的一些个性化配置,以xml文件方式保存在插件同级目录下,名称与插件名一致。

二、插件的父类

所有插件都必须继承自MonitorTask抽象类,这里之所以使用抽象类而不使用接口是考虑到可以在这个抽象类中实现一些共性的东西,如参数配置的加载、错误日志的记录等。

    public abstract class MonitorTask
{
protected XElement _configElement;
public List<Schedule> Schedules;
protected List<MonitorSetting> MonitorSettings;
private bool _isRunning = false; protected MonitorTask()
{
LoadParameters();
} public void Execute()
{
try
{
if (CouldExecute())
{
_isRunning = true;
ExecuteTask();
ExecuteLog();
}
}
catch (Exception e)
{
new LogText(AppDomain.CurrentDomain.BaseDirectory + @"log.log").Write(GetType().Name + "插件执行异常", e.ToString());
}
finally
{
_isRunning = false;
}
} private bool CouldExecute()
{
if (_isRunning || Schedules == null)
{
return false;
}
foreach (var schedule in Schedules)
{
if (schedule.TimeUp())
{
return true;
}
}
return false;
} protected abstract void ExecuteTask(); private void ExecuteLog()
{
if (ConfigManagement.GetConfig().GetAppSetting("ExecuteLog") == "")
{
new LogText(AppDomain.CurrentDomain.BaseDirectory + @"executelog.log").Write(GetType().Name + "插件执行完毕");
}
} protected void LoadParameters()
{
if (File.Exists(GetConfigFile()))
{
_configElement = XElement.Load(GetConfigFile());
LoadXMLParameters();
}
LoadMonitorSettings();
} private string GetConfigFile()
{
return AppDomain.CurrentDomain.BaseDirectory + GetType().Name + ".xml";
} protected virtual void LoadXMLParameters()
{
} protected void LoadMonitorSettings()
{
MonitorSettings = new List<MonitorSetting>();
string sql = "SELECT Name, Value FROM MonitorTaskSetting WHERE MonitorTaskID IN (SELECT ID FROM MonitorTask WHERE Name = '" + GetType().Name + "')";
using (SqlDataReader reader = SqlAssist.Select("SOMP", sql))
{
while (reader.Read())
{
MonitorSettings.Add(new MonitorSetting(reader.GetString(), reader.GetString()));
}
}
}
}

在开发计划任务插件时只需要继承该类,并实现ExecuteTask方法就行了。在这里ConfigManagement.GetConfig().GetAppSetting(string name)、new LogText(string filename).Write(string value)以及SqlAssist.Select(string dbname, string sql)这三个方法都是笔者在开发过程中自行封装的,为了避免重复劳动,一看名字就该知道这些方法的作用吧。

二、任务调度周期

在MonitorTask抽象类中还有一个List<Schedule>,用于保存任务调度的周期,Schedule也是一个抽象类,该抽象类主要由3个方法:

public static List<Schedule> CreateSchedule(XElement element)

该方法根据保存在数据库中的xml数据构建出Schedule。

public abstract bool TimeUp()

由子类实现该方法,判断是否已到达需要调度任务的时间。

public abstract XElement ToXML()

由子类实现该方法,用于将任务调度周期序列化成xml,便于保存到数据库中。

三、管理类

该类主要根据数据库中的配置利用反射构建出相应的计划任务对象,并将其缓存在Dictionary对象里避免多次构建,同时创建线程执行这些插件。

     public static class MonitorTaskManagement
{
private static Dictionary<string, MonitorTask> _tasks = new Dictionary<string, MonitorTask>(); private static MonitorTask GetMonitorTask(string fileName, string taskType, XElement element)
{
Assembly assembly = Assembly.LoadFrom(AppDomain.CurrentDomain.BaseDirectory + fileName);
Type type = assembly.GetType(taskType);
if (type.IsSubclassOf(typeof(MonitorTask)))
{
MonitorTask task = Activator.CreateInstance(type) as MonitorTask;
task.Schedules = Schedule.CreateSchedule(element);
return task;
}
else
{
throw new Exception("程序集" + taskType + "有误。");
}
} private static MonitorTask GetMonitorTask(string taskName, XElement element)
{
MonitorTask task;
if (_tasks.ContainsKey(taskName))
{
task = _tasks[taskName];
}
else
{
task = GetMonitorTask(taskName + ".dll", "MonitorTask." + taskName, element);
_tasks.Add(taskName, task);
}
return task;
} public static void ExecuteTask()
{
string sql = "SELECT Name, Schedule FROM MonitorTask WHERE Enable = 1";
using (SqlDataReader reader = SqlAssist.Select("SOMP", sql))
{
while (reader.Read())
{
try
{
MonitorTask task = GetMonitorTask(reader.GetString(), XElement.Parse(reader.GetString()));
Thread childThread = new Thread(task.Execute);
childThread.IsBackground = true;
childThread.Start();
}
catch (Exception e)
{
new LogText(AppDomain.CurrentDomain.BaseDirectory + @"log.log").Write(reader.GetString() + "插件异常", e.ToString());
}
}
}
}
}

使用C#开发计划任务调度服务的更多相关文章

  1. 【开源】OSharp3.3框架解说系列:开发计划与进度

    OSharp是什么? OSharp是个快速开发框架,但不是一个大而全的包罗万象的框架,严格的说,OSharp中什么都没有实现.与其他大而全的框架最大的不同点,就是OSharp只做抽象封装,不做实现.依 ...

  2. Fragment开发计划

    Fragment是什么 Fragment正如字面意思所言是碎片,所以这是一个管理碎片时间的应用程序.目前考虑的是先在Android上实现,如果IOS的合作伙伴靠谱可以交给他做,如果不靠谱就等Andro ...

  3. 【NetDIY智能主控】开发计划

    经过一个轮回,硬件开发.硬件创业又被推到了历史的前台. 面向低端.初级的硬件爱好者,以Arduino和81单片开发板为核心的开源硬件越来越深入人心,参与的人群越来越多,相关硬件和周边模块也越来越便宜. ...

  4. PHP 开发社区微信服务号实战图解

    本博文就月初刚上线的微信服务号,图文进行总结分享给大家. 去年年底,我所在的团队讨论要开发微信号,话题由此拉开: 原来有一个3年前注册的微信号,但是后台操作无法从“订阅号”变更为“服务号”,随即找腾讯 ...

  5. 开发工具及服务年度大奖评选 I Bugtags 荣获最具成长潜力奖

    作为全球最大中文 IT 社区和服务平台.中国最大技术管理者平台的 CSDN 在中国北京总部举办了一场 2015 年开发工具及服务年度大奖评选活动,此次活动目的在于推动开发服务及工具质量的提升,提高行业 ...

  6. 引擎设计跟踪(九.14.2h) 开发计划

    以后的开发计划: 完善game runtime code, 跑简单的demo目前只有编辑器的运行流程, 没有游戏/demo流程, 图形的测试主要在编辑器上测试, 现在需要测试android系统的图形, ...

  7. 基于SpringBoot开发一个Restful服务,实现增删改查功能

    前言 在去年的时候,在各种渠道中略微的了解了SpringBoot,在开发web项目的时候是如何的方便.快捷.但是当时并没有认真的去学习下,毕竟感觉自己在Struts和SpringMVC都用得不太熟练. ...

  8. Topshelf 一个简化Windows服务开发的宿主服务框架

    Topshelf是 基于.net框架开发的宿主服务框架.该框架简化了服务的创建,开发人员只需要使用 Topshelf编写一个控制台程序,就能安装为Windows服务.之所以这样原因非常简单:调试一个控 ...

  9. Xianfeng轻量级Java中间件平台:一期开发计划

    关于Xianfeng轻量级Java中间件平台,考虑到需要控制开发周期,通过分期开发的方式来实现一些基础的.常用的功能,这样有利于跟踪开发计划.一期的开发计划,主要实现的目标如下: 系统架构: 1.确定 ...

随机推荐

  1. Android Studio的简单设置:

    4.关闭更新: 如下图所示: 6.添加api文档悬浮提示: AS默认是没有api文档悬浮提示的,只有按住[Ctrl+Q]太会出现提示.如果要添加api的自动悬浮提示,设置如下: 上图中,在红框部分打钩 ...

  2. Spark 累加器

    由于spark是分布式的计算,所以使得每个task间不存在共享的变量,而为了实现共享变量spark实现了两种类型 - 累加器与广播变量, 对于其概念与理解可以参考:共享变量(广播变量和累加器).可能需 ...

  3. LeetCode() Range Sum Query-mutable

    http://www.java3z.com/cwbwebhome/article/article1/1369.html?id=4804 http://www.cnblogs.com/zichi/p/4 ...

  4. 为maven插件设置参数的三种方法

    很多的maven插件都提供了丰富的可选参数,用户可以通过设置特定的参数值来控制maven插件的行为.设置插件参数的方法主要有三种,分别是命令行设置,POM文件中为插件设置全局参数和POM文件中为插件设 ...

  5. Myeclipse添加外部Tomcat出现启动故障的问题解决

    故障: 1.java.lang.IllegalStateException: No output folder 分析:work文件夹无写权限 解决:找到tomcat的安装文件夹,右键点击work文件夹 ...

  6. Python之路,day5-Python基础

    for#列表生成式 data = [1,2,3,4,5,6,7] #####列表生成式 #data = [i+1 for i in data] data = [i*2 if i>5 else i ...

  7. SQL 用户定义表类型,在存储过程里使用表类型,表参数作参数

    .定义表类型SUTDENTTYPE,包含三个字段,分别对应学生表的NAME,SEX和PHONE.之所以如此创建,我是准备在插入新学生数据的存储过程中,以它为参数.   GO CREATE TYPE S ...

  8. [Gradle]填坑记录

    1.初次打开Gradle工程特别慢,一直提示下载更新Gradle 解决办法:打开Gradle工程子目录:"\gradle\wrapper" 下的 "gradle-wrap ...

  9. 在CentOS上安装和部署Shiny Server

    1.安装R: sudo yum install R 2.安装Shiny的R包: sudo su - \ -c "R -e \"install.packages('shiny', r ...

  10. 【MVC】ASP.NET MVC Forms验证机制

    http://www.cnblogs.com/bomo/p/3309766.html 随笔 - 121  文章 - 0  评论 - 92 [MVC]ASP.NET MVC Forms验证机制 ASP. ...