在系统运维中常常需要定期去跑一些计划任务,比如扫描服务器监控其性能、检查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. JS实现HashMap

    /** * ********* 操作实例 ************** * var map = new HashMap(); * map.put("key1","Valu ...

  2. HTML5新标签video在iOS上默认全屏播放

    今天做一个app时发现一个问题,应用html5中的video标签加载视频,在Android手机上默认播放大小,但是换成iPhone手机上出问题了,默认弹出全屏播放,查找了好多论坛,都没有谈论这个的.然 ...

  3. 小制作-css+html旋转木马

    源代码: <!DOCTYPE html><html><head>    <title></title>    <meta charse ...

  4. Linux破解root密码

    实验环境                 虚拟机软件:VMware Workstation                 操作系统:Read Hat Enteprise 6.3      1.破解r ...

  5. rabbitmq 重复ACK导致消息丢失

    rabbitmq 重复确认导致消息丢失 背景 rabbitmq 在应用场景中,大多采用工作队列 work-queue的模式. 在一个常见的工作队列模式中,消费者 worker 将不断的轮询从队列中拉取 ...

  6. 039. asp.netWeb用户控件之七实现具有虚拟键盘的功能的用户控件

    用户控件ascx代码: <%@ Control Language="C#" AutoEventWireup="true" CodeFile="K ...

  7. Unexpected end of file from server 服务器访问问题导致

    Caused by: java.net.SocketException: SocketException invoking http://xxxx/cxf/xh/creditInterface?wsd ...

  8. php://input

    从官网信息来看,php://input是一个只读信息流,当请求方式是post的,并且enctype不等于"multipart/form-data"时,可以使用php://input ...

  9. IOS-UITextField键盘不隐藏问题

    这久在手机也页面做了注册功能,需要很多的UITextField,有些是手动输入文字的,有些是点击下拉框选择的,这就出现了当点击下拉框时,就需要将键盘隐藏. - (void)textFieldDidBe ...

  10. switch

    关于java中switch使用的一些说明 switch(表达式) { case常量表达式1:语句1; .... case常量表达式2:语句2; default:语句; } default就是如果没有符 ...