windows service编程
1 基本概念
1.1windows服务简介
创建在它们自己的 Windows 会话中可长时间运行的可执行应用程序。 这些服务可以在计算机启动时自动启动,可以暂停和重新启动而且不显示任何用户界面。
1.2适用情形
适合在服务器上使用且不需要界面的,长时间稳定运行的情形。
1.3如何安装及卸载服务
1)安装
使用InstallUtil.exe命令,在命令提示符工具中执行安装命令。具体如下
首先,执行命令cd C:WindowsMicrosoft.NETFrameworkv4.0.30319
然后,执行InstallUtil.exe path
其中path为待安装服务应用路径。注意,windows server 2012 使用如下安装命令:.InstallUtil.exe path
2)卸载
InstallUtil.exe /u path
注意,windows server 2012 使用如下安装命令:
.InstallUtil.exe /u path
3) 如何管理服务
使用“服务控制管理器”启动、停止、暂停、继续和配置服务,如下图所示:
2 创建服务
2.1四个方法
使用多线程启动、停止、暂停、继续
protected override void OnStart(string[] args):启动
protected override void OnStop():停止
protected override void OnPause():暂停
protected override void OnContinue():继续
2.2 添加安装程序
第一步,双击下图中红框位置。
第二步,在如下的页面中,单击右键。
弹出下图,点击红框位置
第三步,如下界面配置安装程序。
下面两个图的参数含义为:
Account:账户类型。公有四种,入下图
成员名称 |
说明 |
---|---|
LocalService |
一个可用作本地计算机上的非特权用户,并向任意远程服务器提供匿名凭据的帐户。 |
LocalSystem |
一个帐户,使用服务控制管理器,该本地计算机上拥有许多特权并作为网络上的计算机。 |
NetworkService |
提供多种本地特权,并提供给所有远程服务器的计算机的凭据的帐户。 |
User |
定义特定用户在网络上的帐户。 指定 User 为 ServiceProcessInstaller.Account 成员会导致系统提示输入有效的用户名和密码时安装了服务,除非您将值设置为 Username 和 Password 属性您 ServiceProcessInstaller 实例。 |
DelayedAutoStart:该值指示服务是否应推迟之前运行其他自动启动的服务无法启动。true 对延迟自动启动服务;否则为 false。 默认值为 false。
Description :解释服务作用的简短注释。
DisplayName :标识服务的名称。
ServiceName :系统用来标识此服务的名称。
ServicesDependedOn :此服务依赖的服务。
StartType :如何以及何时启动此服务。
成员名称 |
说明 |
---|---|
Automatic |
指示服务将由(或已由)操作系统在系统启动时启动。 如果一个自动启动的服务依赖于手动启动的服务,则该手动启动的服务也会在系统启动时自动启动。 |
Disabled |
指示服务已禁用,因此无法由用户或应用程序启用。 |
Manual |
指示服务仅由用户(使用服务控制管理器)或应用程序以手动方式启动。 |
3 一些技巧
3.1 服务名称配置化
获得配置文件中配置的服务名称,代码如下:
public class ServiceNameSetting
{
/// <summary>
/// 服务名称
/// </summary>
/// <returns></returns>
public static string ServiceName
{
get
{
return GetAppConfigText(GetConfigPath("HYMDService.exe.config"), "ServiceName", "HYMDService");
}
}
/// <summary>
/// 获得配置文件路径
/// </summary>
/// <param name="appConfigName">应用程序配置文件名称</param>
/// <returns></returns>
private static string GetConfigPath(string appConfigName)
{
string root = System.Reflection.Assembly.GetExecutingAssembly().Location;
return root.Remove(root.LastIndexOf('\') + 1) + appConfigName;
}
/// <summary>
/// 获得配置节下内容
/// </summary>
/// <param name="configpath">应用程序配置文件路径</param>
/// <param name="strKeyName"><appSettings>配置节下add节点key属性对应值</param>
/// <param name="defaultName">默认服务名称</param>
/// <returns></returns>
private static string GetAppConfigText(string configpath, string strKeyName, string defaultName)
{
if (!string.IsNullOrWhiteSpace(configpath) && !string.IsNullOrWhiteSpace(strKeyName))
{
using (XmlTextReader tr = new XmlTextReader(configpath))
{
while (tr.Read())
{
if (tr.NodeType == XmlNodeType.Element)
{
if (tr.Name == "add" && tr.GetAttribute("key") == strKeyName)
{
return tr.GetAttribute("value");
}
}
}
}
}
return defaultName;
}
}
注意:
上述代码中不使用System.Configuration.ConfigurationManager.AppSettings[Key]读取配置文件是因为,在程序刚启动时此方式不能读取配置文件,所以要用本文中使用的方法。
3.2 长任务暂停与继续
思路:
在OnPause方法中将FlowController.IsToPause = true,表示服务应该暂停。在OnContinue方法中调用FlowController.Continue方法。
在代码中,需要暂停的位置调用FlowController.Pause,当点击“暂停的时候”,发出暂停命令,程序运行到FlowController.Pause方法时便停在此处。
Pause方法中使用AutoEvent.WaitOne()阻塞线程,直到收到信号。
Continue方法中使用AutoEvent.Set()发出信号,允许线程继续执行。
流程控制器,代码如下:
public class FlowController
{
public static bool IsToPause = false;
public static AutoResetEvent AutoEvent = new AutoResetEvent(false);
/// <summary>
/// 暂停服务
/// </summary>
public static void Pause(params)
{
//对参数params进行处理
//记录程序暂停了
AutoEvent.WaitOne();
}
/// <summary>
/// 服务继续
/// </summary>
public static void Continue(params)
{
//对参数params进行处理
//记录程序继续了
AutoEvent.Set();
IsToPause = false;
}
}
-----------------------------------------------------------------------------------------
时间仓促,水平有限,如有不当之处,欢迎指正。
- 【Dev Club 分享】微信 iOS SQLite 源码优化实践
- 移动客户端中高效使用 SQLite
- 【Dev Club 分享】微信热补丁 Tinker 的实践演进之路
- Android 进程保活招式大全
- 【Dev Club 分享】H5 视频直播那些事
- Android Patch 方案与持续交付
- Linux系统下MongoDB的简单安装与基本操作
- Go语言同步(Synchronization)
- 服务器反爬虫攻略:Apache/Nginx/PHP禁止某些User Agent抓取网站
- 【Dev Club分享】JSPatch成长之路
- Node.js新手必须知道的4个JavaScript概念
- 博客文章重新启用评论,附一键填写评论中用户信息代码生成工具
- 提升 Node.js 应用性能的 5 个技巧
- 再次扩散:Linux系统bash漏洞CVE-2014-6271仍未被彻底修复,红帽再发补丁
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- A little fun with InnoDB multi-versioning(14.有关InnoDB多版本中的一个小问题)
- 聊聊java中的哪些Map:(八)ConcurrentSkipListMap源码分析
- littlevgl(Lvgl)最新版V7.4移植
- TiKV 源码解析系列文章(二十)Region Split 源码解析
- 轻松构建Tomcat源码
- Flutter中Contrainer 组件的宽高限制分析
- 10张图带你深入理解Docker容器和镜像
- 手摸手教你撸一个微服务框架-关于服务端的处理
- 聊聊claudb的string command
- windows下安装nodejs
- 【Java面试总结】Java集合
- 《JavaScript 模式》读书笔记(8)— DOM和浏览器模式1
- 《JavaScript 模式》读书笔记(8)— DOM和浏览器模式2
- 5000字 | 24张图带你彻底理解21种并发锁
- JavaScript-变量