windows服务插件利器-新生命组件XAgent使用心得
1.简单介绍
XAgent为大石头带领下的新生命团队自己开发的一个.Net下的常用的Windows服务管理组件利器,通过在控制台中简单的输入1,2,3,4,5等数字可以实现一步安装、卸载WindowsService以及单步调试和循环调试、服务停止,重启、运行,用起来很方便。看效果(具体使用方法可以去XAgent官网查看):

2.当前使用场景介绍及实现
我的这个项目下面有很多控制台应用程序需要运行,平时部署很麻烦,不方便管理。我的想法是把这些程序以插件的形式放到一个应用程序中运行管理。这里写了一个Ijob接口。看代码:
Ijob相关代码
/// <summary>工作接口</summary>
public interface IJob
{
/// <summary>
/// 工作组件顺序
/// </summary>
int Sort { get; set; } /// <summary>
/// 间隔时间
/// </summary>
int JobInterval { get; set; } /// <summary>启动</summary>
/// <returns></returns>
bool Start(); /// <summary>停止</summary>
/// <returns></returns>
bool Stop(); /// <summary>工作</summary>
bool Work(); //void WriteLog(string format, params object[] args);
} /// <summary>基础工作组件</summary>
public class Job : IJob
{
///// <summary>
///// 是否执行
///// </summary>
//public bool IsExcute { get; set; } public Job()
{
Sort = ;
DisplayName = "基础工作组件";
JobInterval = ; //默认60秒
//IsExcute = true;//默认执行
} /// <summary>工作组件名</summary>
public virtual string DisplayName { get; set; } /// <summary>工作组件顺序</summary>
public virtual int Sort { get; set; } /// <summary>
/// 时间间隔
/// </summary>
public int JobInterval { get; set; } /// <summary>启动</summary>
public virtual bool Start()
{
WriteLog("开始工作");
return false;
} /// <summary>停止</summary>
/// <returns></returns>
public virtual bool Stop()
{
WriteLog("停止工作");
return false;
} /// <summary>工作</summary>
public virtual bool Work()
{
WriteLog("正在运行");
return false;
} public virtual void WriteLog(string format, params object[] args)
{
//XTrace.WriteLine("[" + this.GetType() + "] [" + DisplayName + "] " + format, args);
XTrace.WriteLine("[" + DisplayName + "] " + format, args);
}
public static class JobHelper
{
static readonly List<Type> _hasInited = new List<Type>(); private static List<IJob> op_cache = new List<IJob>(); /// <summary>所有工作组件</summary>
public static List<IJob> Works
{
get
{
var list = LoadWork();
foreach (var item in list)
{
EnsureInit(item);
}
op_cache = op_cache.OrderBy(j => j.Sort).ToList();
return op_cache;
}
} /// <summary>确保实体类已经执行完静态构造函数,因为那里实在是太容易导致死锁了</summary>
/// <param name="type">类型</param>
internal static void EnsureInit(Type type)
{
if (_hasInited.Contains(type)) return;
// 先实例化,在锁里面添加到列表但不实例化,避免实体类的实例化过程中访问CreateOperate导致死锁产生
var item = type.CreateInstance();
lock (_hasInited)
// 如果这里锁定_hasInited,还是有可能死锁,因为可能实体类A的静态构造函数中可能导致调用另一个实体类的EnsureInit
// 其实我们这里加锁的目的,本来就是为了避免重复添加同一个type而已
//lock ("_hasInited" + type.FullName)
{
if (_hasInited.Contains(type)) return; if (op_cache.Contains(item as IJob)) return;
op_cache.Add(item as IJob); //type.CreateInstance();
_hasInited.Add(type);
}
} /// <summary>列出所有工作接口</summary>
/// <returns></returns>
public static List<Type> LoadWork()
{
return typeof (IJob).GetAllSubclasses(true).ToList();
}
}
确保所有的工作组件继承Job类,重写其中的方法,将一些需要配置的属性做成xml进行管理。废话不说看代码:
这是其中一个工作插件
public HourStatisticsWork()
{ DisplayName = "小时统计组件";
Sort = ; if (WorkSetting.Current.HourStatisticsEnable)
{
JobInterval = WorkSetting.Current.HourStatisticsInterval;
}
else
{
JobInterval = -; //禁用此组件
}
} public override bool Work()
{
StartWork();
return base.Work();
}
public override bool Stop()
{
if (_HourTimer != null)
{
_HourTimer.Dispose();
}
return base.Stop();
}
public void StartWork()
{
try
{
// Console.WriteLine("当前时间{0}",DateTime.Now);
//var time = WorkSetting.Current.HourStatisticsInterval;
//// 定时器一分钟后启动
//if (_HourTimer == null)
// _HourTimer = new TimerX(obj =>
// { // StartWork();
// }, null, 3 * 1000, time);
DateTime dt = DateTime.Now.AddHours(-);
cyStatistics(dt);
sccStatistics(dt);
ccStatistics(dt);
jcStatistics(dt);
mfStatistics(dt);
vcStatistics(dt);
vbStatistics(dt);
pulStatistics(dt);
icStatistics(dt);
rhcStatistics(dt);
dtrStatistics(dt); MotorLoadStatistics(dt);
TimeStatistics(dt);
Container.Info("[HourStatisticsWork]Hour Statistics");
}
catch (Exception ex)
{
Container.Error($"[HourStatisticsWork]{ex.Message}");
} }
相关配置项
[DisplayName("运行参数设置")]
[XmlConfigFile(@"Config\WorkSetting.config", )]
public class WorkSetting : XmlConfig<WorkSetting>
{
//private String _LastTool;
///// <summary>最后一个使用的工具</summary>
//[DisplayName("最后一个使用的工具")]
//public String LastTool { get { return _LastTool; } set { _LastTool = value; } }
protected override void OnNew()
{
SystemName = "物联网监控系统";
LastUpdateTime = new DateTime(, , );
RabbitMqResolveEnable = true;
RabbitMqResolveInterval = ;//默认1s
HourStatisticsEnable = true;
HourStatisticsInterval = *;//默认30min
DayStatisticsEnable = true;
DayStatisticsInterval = * *;//默认12个小时
EnableViewServiceLog = false;
}
#region 队列解析
[Description("启用队列解析")]
public bool RabbitMqResolveEnable { get; set; }
[Description("队列解析间隔时间,单位:秒")]
public int RabbitMqResolveInterval { get; set; }
#endregion
#region 小时统计
[Description("启用小时统计")]
public bool HourStatisticsEnable { get; set; }
[Description("小时统计间隔时间,单位:秒")]
public int HourStatisticsInterval { get; set; }
#endregion
#region 自然日统计
[Description("启用自然日统计")]
public bool DayStatisticsEnable { get; set; }
[Description("自然日统计间隔时间,单位:秒")]
public int DayStatisticsInterval { get; set; }
#endregion
#region 异常通知
[Description("启用异常通知")]
public bool ExceptionNotifyEnable { get; set; }
[Description("异常通知级别")]
public int ExceptionNotifyLevel { get; set; }
#endregion 异常通知
#region 系统
/// <summary>
/// 系统名称
/// </summary>
[Description("系统名称")]
public String SystemName { get; set; }
/// <summary>最后更新时间</summary>
[Description("最后更新时间")]
public DateTime LastUpdateTime { get; set; }
#endregion 系统
#region Windows Service
/// <summary>服务名</summary>
[Description("服务名")]
public String ServiceName { get; set; }
/// <summary>显示名</summary>
[Description("显示名")]
public String DisplayName { get; set; }
/// <summary>服务描述</summary>
[Description("服务描述")]
public String Description { get; set; }
/// <summary>查看服务运行日志</summary>
[Description("查看服务运行日志")]
public bool EnableViewServiceLog { get; set; }
#endregion Windows Service
}
一切准备好后,开始生成项目并运行,效果如下:

3.发现问题
看起来一切运行正常,但是等到我讲xml文件中的HourStatisticsEnable设置成false时,发现只有第一个队列解析的服务在运行。第三个自然日统计的服务没有运行。这时我注意到,原来是插件运行的先后顺序影响的,即Sort属性。本质上还是JobInterval控制了改插件是否运行。可能XAgent的源代码有些小问题,我没有继续深入到源代码中去查找解决,而是直接通过xml配置文件的Enable判断该组件是否运行。
代码如下:
public HourStatisticsWork()
{ if (WorkSetting.Current.HourStatisticsEnable)
{
DisplayName = "小时统计组件";
Sort = ;
JobInterval = WorkSetting.Current.HourStatisticsInterval;
}
else
{
//JobInterval = -1; //禁用此组件
}
} public override bool Work()
{
if (WorkSetting.Current.HourStatisticsEnable)
{
StartWork();
}
return base.Work();
}
public override bool Stop()
{
if (_HourTimer != null)
{
_HourTimer.Dispose();
}
return base.Stop();
}
public void StartWork()
{//todo}
另外,该IJob接口借鉴了我曾经的领导的想法。
供大家学习,如有不正不妥之处请大家指正。谢谢!
需要源码的或者帮助的联系我QQ:694666781
windows服务插件利器-新生命组件XAgent使用心得的更多相关文章
- 新生命组件XAgent使用心得
1.简单介绍 XAgent为大石头带领下的新生命团队自己开发的一个.Net下的常用的Windows服务管理组件利器,通过在控制台中简单的输入1,2,3,4,5等数字可以实现一步安装.卸载Windows ...
- 【总结】探索Newlife组件:服务代理利器XAgent的前世今生
本博客所有文章分类的总目录:http://www.cnblogs.com/asxinyu/p/4288836.html Newlife XCode组件相关文章目录:http://www.cn ...
- 配置jboss为windows服务
先确保jdk和jboss的环境变量是正常可用的 1.(下载binaries 2.x.x-windows x86)找到service.bat和jbosssvc.exe两个文件 1.1 binaries ...
- 制作Windows服务和安装程序(C#版)
http://blog.sina.com.cn/s/blog_5f4ffa170100vt2b.html 1.创建服务项目: 打开VS 2005 编程环境,在C#中新建Windows服务程序 2.将安 ...
- 使用 Topshelf 组件一步一步创建 Windows 服务 (2) 使用Quartz.net 调度
上一篇说了如何使用 Topshelf 组件快速创建Windows服务,接下来介绍如何使用 Quartz.net 关于Quartz.net的好处,网上搜索都是一大把一大把的,我就不再多介绍. 先介绍需要 ...
- 使用Topshelf组件构建简单的Windows服务
很多时候都在讨论是否需要了解一个组件或者一个语言的底层原理这个问题,其实我个人觉得,对于这个问题,每个人都有自己的看法,个人情况不同,选择的方式也就会不同了.我个人觉得无论学习什么,都应该尝试着去了解 ...
- js replace 全局替换 以表单的方式提交参数 判断是否为ie浏览器 将jquery.qqFace.js表情转换成微信的字符码 手机端省市区联动 新字体引用本地运行可以获得,放到服务器上报404 C#提取html中的汉字 MVC几种找不到资源的解决方式 使用Windows服务定时去执行一个方法的三种方式
js replace 全局替换 js 的replace 默认替换只替换第一个匹配的字符,如果字符串有超过两个以上的对应字符就无法进行替换,这时候就要进行一点操作,进行全部替换. <scrip ...
- Windows Service插件服务开源
WindowsService 插件服务是一个为简化NTService开发和打包程序,提供插件开发的方式进行动态加入或删除业务. 插件式服务程序的由来,在系统维护的过程中,根据企业的要求经常要进行一些周 ...
- C# 使用Vici WinService组件来创建Windows服务
Vici WinService 是 Windows平台下使用C#开发的轻量级用于创建,删除服务的类库,您只需简单的几行代码即可实现多线程异步服务的创建,删除,运行 废话不多说,直接上代码 /***** ...
随机推荐
- LSTM入门学习——本质上就是比RNN的隐藏层公式稍微复杂了一点点而已
LSTM入门学习 摘自:http://blog.csdn.net/hjimce/article/details/51234311 下面先给出LSTM的网络结构图: 看到网络结构图好像很复杂的样子,其实 ...
- 111.final与override
#include <iostream> using namespace std; class myclass { public: //后面加一个final,则禁止虚函数被子类重写 //fi ...
- 2015合肥网络赛 HDU 5492 Find a path 动归
HDU 5492 Find a path 题意:给你一个矩阵求一个路径使得 最小. 思路: 方法一:数据特别小,直接枚举权值和(n + m - 1) * aver,更新答案. 方法二:用f[i][j] ...
- mysql-5.6.15 开启二进制文件
windows下 mysql 开启二进制文件 在mysql5.6.15下存在 my-default.ini配置文件 复制新建重命名my.ini 在其下加入 一定要在 [mysqld] 下面添加, ...
- TortoiseSvn介绍 客户端
转载自:http://www.cnblogs.com/lyhabc/articles/2482381.html TortoiseSvn 是 Subversion 版本控制系统的一个免费开源客户端,可以 ...
- vue踩坑-This dependency was not found
* vux in ./node_modules/babel-loader/lib!./node_modules/vue-loader/lib/selector.js?type=script&i ...
- IDEA 热启动,每次更改代码后不用重启服务
1.ctrl+Shift+Alt+/,选择Registry 2.勾选 compiler.automake.allow.when.app.running(可能不按首字母排序,可以多找找) 3.Setti ...
- 游戏server之server优化思路
本文仅仅是提供一些游戏server优化思路,当中一些思路是用在不同场合的,不是同个架构的.须要依据应用场景选用合适方式. 本文的引用的文章都是在自己写的在本博客内的.也都是上线开几百个服的成熟项目的. ...
- Bmob移动后端云服务平台--Android从零開始--(一)何为Bmob
Bmob移动后端云服务平台--Android从零開始--(一)何为Bmob 在正式的项目开发中,单client不能满足我们的需求,须要实现client与服务端的连接. 而在编写Android服务端代码 ...
- css 浮动 绝对定位 和 相对定位
html是按照文件流(文档流)的方式加载的,但是全部是文档流的话,很多好看的样式是实现不了的,所以出现了浮动,相对定位,绝对定位的概念. 一.首先,按照文档流和非文档流来分类: ①文档流:就是按照上下 ...