使用Topshelf组件 一步一步创建 Windows 服务
我们先来介绍一下使用它的好处,以下论述参考自其他大神。
topshelf是创建windows服务的一种方式,相比原生实现ServiceBase、Install.Installer更为简单方便, 我们只需要几行代码即可实现windows服务的开发。
topshelf本身支持windows及linux下mono上部署安装,同样也是开源的。
topshelf相对原生来说,调试起来比较方便,可以在开发时以控制台的形式直接f5调试,发布时用命令以服务的形式部署。
还一个比较有用的特性是支持多实例的部署,这样可以在一台机器上部署多个相对的服务。类似的工具有instsrv和srvany。
多实例有一个好处就是容灾,当一个服务部署多份时,这样其中任何一个服务实例挂了,剩余的可以继续执行。
多实例可以是主备的方式,主挂了备服务才会执行。也可以以负载均衡的方式实现,多实例抢占进程锁或分布式锁,谁拿到谁执行。
先写出具体步骤:
// 新建控制台应用程序
// 使用Nuget安装Topshelf,选择能用的最新版本
// 使用Nuget安装NLog和NLog.config,选择能用的最新版本,用于打印日志 Nlog需要配置文件,详见NLog.config
// 初始化配置文件,创建AppConfigHelper类,继承 ConfigurationSection (需要引用System.Configuration程序集)
// 完善App.Config配置文件,读取App.Config配置文件,具体查看AppConfigHelper类
// 创建一个注册服务类TopshelfRegistService,初始化Topshelf注册
// 我们的目标很简单,就是让服务打印一个日志文件
// 编译并生成项目,进入 bin\Debug 目录下,找到xxx.exe 执行 install 命令,Windows 服务就诞生了
// 注意:如果出现需要以管理员身份启动的提示,重新以管理员身份启动 cmd
//接下来直接上代码与截图



卸载服务:

当我们启动服务的时候,成功打印出了日志,表示一切成功

程序结构很简单,如下图所示:

接下来,我们直接上实现代码,我会按照步骤依次给出:
1,Program主程序代码
namespace ProcessPrintLogService
{
class Program
{
public static readonly Logger log = LogManager.GetCurrentClassLogger();
private static readonly AppConfigHelper config = AppConfigHelper.Initity();
static void Main(string[] args)
{
TopshelfRegistService.Regist(config, true);
}
}
}
2.AppConfigHelper类,用于读取配置文件,使用配置文件的方式可以使你后期将该服务应用于多个应用程序
namespace ProcessPrintLogService
{
public class AppConfigHelper : ConfigurationSection
{
private static AppConfigHelper _AppConfig = null;
private static readonly object LockThis = new object(); /// <summary>
/// 获取当前配置 获取section节点的内容
/// 使用单例模式
/// </summary>
/// <returns></returns>
public static AppConfigHelper Initity()
{
if (_AppConfig == null)
{
lock (LockThis)
{
if (_AppConfig == null)
{
//获取app.config文件中的section配置节点
_AppConfig = (AppConfigHelper)ConfigurationManager.GetSection("AppConfigHelper");
}
}
}
return _AppConfig;
} //创建一个AppConfigHelper节点
//属性分别为:ServiceName、Desc 等....
//这里介绍一下属性标签:ConfigurationProperty 它可以在配置文件中根据属性名获取Value值
//可以参考文章https://www.cnblogs.com/liunlls/p/configuration.html /// <summary>
/// 服务名称
/// </summary>
[ConfigurationProperty("ServiceName", IsRequired = true)]
public string ServiceName
{
get { return base["ServiceName"].ToString(); }
internal set { base["ServiceName"] = value; }
} /// <summary>
/// 描述
/// </summary>
[ConfigurationProperty("Desc", IsRequired = true)]
public string Description
{
get { return base["Desc"].ToString(); }
internal set { base["Desc"] = value; }
} }
}
3.Topshelf组件注册服务
namespace ProcessPrintLogService
{
/// <summary>
/// Topshelf组件注册服务
/// </summary>
internal class TopshelfRegistService
{
/// <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; //默认不注册 //服务用本地系统账号来运行
x.RunAsLocalSystem();
//服务的描述信息
x.SetDescription(config.Description);
//服务的显示名称
x.SetDisplayName(config.ServiceName);
//服务的名称(最好不要包含空格或者有空格属性的字符)Windows 服务名称不能重复。
x.SetServiceName(config.ServiceName);
});
host.Run(); //启动服务 如果使用HostFactory.Run()则不需要该方法
}
} /// <summary>
/// 自定义服务
/// </summary>
internal class QuartzHost
{
public readonly Logger log = LogManager.GetLogger("QuartzHost"); public QuartzHost()
{
var service = AppConfigHelper.Initity();
} //服务开始
public void Start()
{
try
{
Task.Run(() =>
{
log.Info($"服务开始成功!");
});
}
catch (Exception ex)
{
Task.Run(() =>
{
log.Fatal(ex, $"服务开始失败!错误信息:{0}", ex);
});
throw;
}
} //服务停止
public void Stop()
{
Task.Run(() =>
{
log.Trace("服务结束工作");
});
}
} }
4.App.config配置文件
<?xml version="1.0" encoding="utf-8" ?>
<configuration> <!--该节点一定要放在最上边-->
<configSections>
<section name="AppConfigHelper" type="ProcessPrintLogService.AppConfigHelper,ProcessPrintLogService"/>
</configSections> <!--TopSelf服务配置文件 -->
<AppConfigHelper
ServiceName="Process_PrintLogService"
Desc="日志打印服务"
/> <!--数据库连接字符串 -->
<connectionStrings>
<add name="ConnectionString" connectionString=""/>
</connectionStrings> <startup>
<supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5.2" />
</startup>
</configuration>
5.Nlog.config日志配置文件
<?xml version="1.0" encoding="utf-8" ?>
<nlog xmlns="http://www.nlog-project.org/schemas/NLog.xsd"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<targets>
<!--type="File|Console" 属性是设置日志输出目标是"File"(文件)或者"Console"(控制台)-->
<!--fileName="${basedir}/logs/${shortdate}/${level}/${callsite}.log" 设置日记记录文件的路径和名称-->
<!--layout="${longdate} ${level} ${callsite}:${message}" 设置日志输出格式-->
<target name="t1"
type="File"
fileName="${basedir}/logs/${shortdate}/${level} ${callsite}.log"
layout="${longdate} ${level} ${callsite}:${message}"
archiveAboveSize=""
archiveNumbering="Rolling"
concurrentWrites="false"
keepFileOpen="true"
maxArchiveFiles =""
/> <!--输出至控制台-->
<target name="t2" type="Console" layout="${longdate} ${level} ${callsite}:${message}" />
</targets> <rules>
<!--如果填*,则表示所有的Logger都运用这个规则,将所有级别的日志信息都写入到“t1”和“t2”这两个目标里-->
<logger name="*" writeTo="t1,t2"/>
</rules>
</nlog>
以上就是此次示例的全部代码,到此你也许会有一个问题,就是我想定时执行我的任务?比如每天几点执行,或者每几分钟执行一次等等,那我们该怎么做呢?
答案是使用:Quartz.net ,接下来我将会使用 Quartz.net 实现上述的定时任务。
参考文献:
https://www.jianshu.com/p/f2365e7b439c
http://www.80iter.com/blog/1451523192435464/
https://www.itsvse.com/thread-7503-1-1.html?tdsourcetag=s_pctim_aiomsg
https://www.cnblogs.com/yanglang/p/7199913.html
使用Topshelf组件 一步一步创建 Windows 服务的更多相关文章
- 使用 Topshelf 组件一步一步创建 Windows 服务 (2) 使用Quartz.net 调度
上一篇说了如何使用 Topshelf 组件快速创建Windows服务,接下来介绍如何使用 Quartz.net 关于Quartz.net的好处,网上搜索都是一大把一大把的,我就不再多介绍. 先介绍需要 ...
- 使用Topshelf 5步创建Windows 服务 z
使用Topshelf创建Windows 服务简要的介绍了创建Windows服务的另一种方法,老外的一篇文章Create a .NET Windows Service in 5 steps with T ...
- 使用Topshelf 5步创建Windows 服务
使用Topshelf创建Windows 服务简要的介绍了创建Windows服务的另一种方法,老外的一篇文章Create a .NET Windows Service in 5 steps with T ...
- C# 使用Vici WinService组件来创建Windows服务
Vici WinService 是 Windows平台下使用C#开发的轻量级用于创建,删除服务的类库,您只需简单的几行代码即可实现多线程异步服务的创建,删除,运行 废话不多说,直接上代码 /***** ...
- 使用Topshelf创建Windows服务
概述 Topshelf是创建Windows服务的另一种方法,老外的一篇文章Create a .NET Windows Service in 5 steps with Topshelf通过5个步骤详细的 ...
- [Solution] Microsoft Windows 服务(2) 使用Topshelf创建Windows服务
除了通过.net提供的windows服务模板外,Topshelf是创建Windows服务的另一种方法. 官网教程:http://docs.topshelf-project.com/en/latest/ ...
- Topshelf创建Windows服务
使用Topshelf创建Windows服务 概述 Topshelf是创建Windows服务的另一种方法,老外的一篇文章Create a .NET Windows Service in 5 steps ...
- 使用 Topshelf 结合 Quartz.NET 创建 Windows 服务
Ø 前言 之前一篇文章已经介绍了,如何使用 Topshelf 创建 Windows 服务.当时提到还缺少一个任务调度框架,就是 Quartz.NET.而本文就展开对 Quartz.NET 的研究,以 ...
- 使用 Topshelf 创建 Windows 服务
Ø 前言 C# 创建 Windows 服务的方式有很多种,Topshelf 就是其中一种方式,而且使用起来比较简单.下面使用 Visual Studio Ultimate 2013 演示一下具体的使 ...
随机推荐
- 安装VMware Tools显示灰色正确解决办法
首先问题如下: 解决办法如下:1.关闭虚拟机: 2.在虚拟机设置分别设置CD/DVD.CD/DVD2和软盘为自动检测三个步骤: 3.再重启虚拟机,灰色字即点亮. 大功告成,如果解决了你的问题,点个赞鼓 ...
- [JZOJ5817] 【NOIP提高A组模拟2018.8.15】 抄代码
Description J 君是机房的红太阳,每次模拟她总是 AK 虐场.然而在 NOIP2117 中,居然出现了另一位 AK 的选手 C 君! 这引起了组委会的怀疑,组委会认为 C 君有抄袭 J 君 ...
- Python小工具:利用ffmpy3库3秒钟将视频转换为音频
作者 | pk 哥 来源公众号 | Python知识圈(ID:PythonCircle) 最近,有读者微信上私聊我,想让我写一篇视频批量转换成音频的文章,我答应了,周末宅家里把这个小工具做出来了. 这 ...
- Cassandra官方介绍及安装
Cassandra官方介绍及安装 这个数据库有很高的伸缩性和可用性,你完全不需要向性能妥协.在硬件或者云设施上做修改 官网:Cassandra官网 一.简介 cassandra这个数据库有很高的伸缩性 ...
- Node.js事件的正确使用方法
前言 事件驱动的编程变得流行之前,在程序内部进行通信的标准方法非常简单:如果一个组件想要向另外一个发送消息,只是显式地调用了那个组件上的方法.但是在 react 中用的却是事件驱动而不是调用. 事件的 ...
- 子字符串查找之————关于KMP算法你不知道的事
写在前面: (阅读本文前需要了解KMP算法的基本思路.另外,本着大道至简的思想,本文的所有例子都会做从头到尾的讲解) 作者翻阅了大量网上现有的KMP算法博客,发现广为流传的竟然是一种不完整的KMP算法 ...
- R的安装
更新时间:2019.09.23 1. 序言 之前曾经用过一段时间的R(一直忍受着原生R那个超级"简洁"的界面),但是后来重装了系统并且学习了Python,就没有再怎么碰过R了.然而 ...
- KAFKA集群搭建(自带zookeeper)
1. KAFKA下载地址:http://kafka.apache.org/downloads KAFKA-快速上手-官方网站:http://kafka.apache.org/quickstart 2. ...
- 源码剖析Yii错误 Invalid parameter number: no parameters were bound
ActiveRecord使用的一个陷阱导致 Invalid parameter number: no parameters were bound 请看下面的例子 $criteria = new CDb ...
- Java多线程编程(四)Lock的使用
一.使用ReentrantLock类 在Java多线程中,可以使用synchronized关键字来实现线程之间的同步互斥,但ReentrantLock类也能达到同样的效果,并且在扩展功能上也更加强大, ...