C#开发Windows服务详细流程
1.Windows服务简单介绍
Windows服务程序是在Windows操作系统下能完成特定功能的可执行的应用程序,主要用于长时间运行的功能或者执行定时任务。一般情况下,用户不能通过用户界面来安装和启动Windows服务程序,服务的启动方式有自动启动和手动启动两种。我们可以在运行中输入services.msc 命令打开服务管理界面,在这里可以进行启动或者停止服务:

2.C#开发windows服务
1.windows服务开发流程
1.1 windows的基本开发流程
使用.Net可以很方便地开发windows服务,下边通过一个栗子演示下开发windows服务的简单流程。
首先创建一个Windows服务项目,如下图我们创建了一个叫MyService的服务项目

然后打开自动生成的Service1.cs,该文件默认有两个方法: OnStart()和OnStop() ,OnStart()指定当服务启动时要执行的操作,OnStop()指定当服务停止时要执行的操作。 我们重写OnStart()方法,让服务开启的时候,每隔一秒在一个文本文件中记录当前时间,代码如下图

然后在右键Service1.cs文件,选择查看设计器,在界面上右键选择添加安装程序,如下图

修改serviceInstaller和serviceProcessInstaller,如下图所示,到这里服务的创建工作就完成了


1.2 安装和卸载Windows服务项目
我们先通过下边命令来安装和卸载服务,其中第二行命令用于安装服务,命令的地址是我们服务项目生成的exe文件所在路径,第三行命令用于卸载服务。
cd C:\Windows\Microsoft.NET\Framework\v4.0.30319 InstallUtil C:\Users\ZDZN\source\repos\WindowsServiceDemo\WindowsServiceInstall\bin\Debug\MyService.exe ----------安装服务
InstallUtil -u C:\Users\ZDZN\source\repos\WindowsServiceDemo\WindowsServiceInstall\bin\Debug\MyService.exe ----------卸载服务
执行成功后在服务管理界面可以看到我们刚才安装的服务,这说明服务已经安装成功了:

我们可以在命令行中通过以下命令控制服务的开启和停止:
net start MyService //开启服务
net stop MyService //停止服务
sc delete MyService //删除服务
如果程序没有错误的话,服务启动后会在F盘中生成一个hello.txt文件,内容如下:

2.通过Winform控制windows服务
虽然通过命令行可以实现Windows服务的安装、卸载、启动、停止等操作,但是这样对于用户来说还是太过麻烦。如果能通过Winform来实现这些功能,用户使用起来就方便多了。怎么让Winform控制windows服务呢?
首先要知道的一些知识:使用AssemblyInstaller可以进行Windows服务的安装和卸载,ServiceController可以控制windows服务的启动,停止,暂停,继续等,知道了这些就容易实现了。
添加一个winform应用程序,引用我们的服务项目,程序代码如下,这里的命名都很明显就不详细说明:
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
//设置服务地址和服务名
string serviceFilePath = $"{Application.StartupPath}/MyService.exe";
string serviceName = "MyService"; private void btnInstall_Click(object sender, EventArgs e)
{
Task.Run(() =>
{
ShowMessage("开始安装服务...");
if (IsServiceExisted(serviceName))
{
UninstallService(serviceFilePath);
}
InstallService(serviceFilePath);
});
} private void btnStart_Click(object sender, EventArgs e)
{
Task.Run(() =>
{
ShowMessage("服务正在启动...");
if (IsServiceExisted(serviceName))
{
ServiceStart(serviceName);
ShowMessage("服务已启动");
} });
} private void btnStop_Click(object sender, EventArgs e)
{
Task.Run(() =>
{
ShowMessage("服务正在停止...");
if (IsServiceExisted(serviceName))
{
ServiceStop(serviceName);
ShowMessage("服务已停止");
}
});
} private void btnUninstall_Click(object sender, EventArgs e)
{
Task.Run(() =>
{
ShowMessage("开始卸载服务...");
if (this.IsServiceExisted(serviceName))
{
this.ServiceStop(serviceName);
}
this.UninstallService(serviceFilePath);
});
} #region 方法封装
private void ShowMessage(string s)
{
Action setState = () =>
{
myTxtBox.AppendText(DateTime.Now.ToString("f") + ":");
myTxtBox.AppendText(Environment.NewLine);
myTxtBox.AppendText(s);
myTxtBox.AppendText(Environment.NewLine); };
myTxtBox.Invoke(setState);
} //判断服务是否存在
private bool IsServiceExisted(string serviceName)
{
try
{
ServiceController[] services = ServiceController.GetServices();
foreach (ServiceController sc in services)
{
if (sc.ServiceName.ToLower() == serviceName.ToLower())
{
return true;
}
}
return false;
}
catch (Exception ex)
{
ShowMessage(ex.Message);
return false;
}
} //安装服务
private void InstallService(string serviceFilePath)
{
try
{
using (AssemblyInstaller installer = new AssemblyInstaller())
{
installer.UseNewContext = true;
installer.Path = serviceFilePath;
IDictionary savedState = new Hashtable();
installer.Install(savedState);
installer.Commit(savedState);
ShowMessage("服务安装完成");
}
}
catch (Exception ex)
{
ShowMessage(ex.Message);
}
} //卸载服务
private void UninstallService(string serviceFilePath)
{
try
{
using (AssemblyInstaller installer = new AssemblyInstaller())
{
installer.UseNewContext = true;
installer.Path = serviceFilePath;
installer.Uninstall(null);
ShowMessage("服务卸载完成");
}
}
catch (Exception ex)
{
ShowMessage(ex.Message);
}
} //启动服务
private void ServiceStart(string serviceName)
{
try
{
using (ServiceController control = new ServiceController(serviceName))
{
if (control.Status == ServiceControllerStatus.Stopped)
{
control.Start();
}
}
}
catch (Exception ex)
{
ShowMessage(ex.Message);
}
} //停止服务
private void ServiceStop(string serviceName)
{
try
{
using (ServiceController control = new ServiceController(serviceName))
{
if (control.Status == ServiceControllerStatus.Running)
{
control.Stop();
}
}
}
catch (Exception ex)
{
ShowMessage(ex.Message);
}
}
#endregion #region 页面元素 private void Form1_Load(object sender, EventArgs e)
{
Control.CheckForIllegalCrossThreadCalls = false; timer1.Interval = ;
timer1.Enabled = true;
} //btn状态
private void ButtonState()
{
if (IsServiceExisted(serviceName))
{
btnInstall.Enabled = false;
btnUninstall.Enabled = true;
using (var service = new ServiceController(serviceName))
{
if (service.Status == ServiceControllerStatus.Running || service.Status == ServiceControllerStatus.StartPending)
{
btnUninstall.Enabled = false;
btnStart.Enabled = false;
btnStop.Enabled = true;
}
else
{
btnUninstall.Enabled = true;
btnStart.Enabled = true;
btnStop.Enabled = false;
}
}
}
else
{
btnInstall.Enabled = true;
btnUninstall.Enabled = false;
btnStart.Enabled = false;
btnStop.Enabled = false;
}
}
//底部label状态
private void LabelState()
{
if (!IsServiceExisted(serviceName))
{
this.labState.Text = $"【{serviceName}】服务未安装";
return;
}
using (var service = new ServiceController(serviceName))
{
switch (service.Status)
{
case ServiceControllerStatus.Running:
this.labState.Text = $"【{serviceName}】服务已启动";
break;
case ServiceControllerStatus.StartPending:
this.labState.Text = $"【{serviceName}】服务正在启动...";
break;
case ServiceControllerStatus.Stopped:
this.labState.Text = $"【{serviceName}】服务已停止";
break;
case ServiceControllerStatus.StopPending:
this.labState.Text = $"【{serviceName}】服务正在停止...";
break;
default:
break;
}
}
} private void timer1_Tick(object sender, EventArgs e)
{
ButtonState();
LabelState();
}
#endregion
}

完成这一步后,我们就可以通过可视化界面对服务的安装,卸载,运行,停止进行控制了。
补充:使用TopShelf也可以实现Windows服务的快速开发,参考文档 http://docs.topshelf-project.com/en/latest/index.html
C#开发Windows服务详细流程的更多相关文章
- C#开发Windows服务 附简单实例实现禁止QQ运行
本实例主要实现下面三个基本功能 1.C#开发windows服务 2.禁止QQ等程序运行 3.为windows服务创建自动安装程序 下面针对这三个基本功能进行实现 一.C#开发windows服务 Win ...
- C# DateTime的11种构造函数 [Abp 源码分析]十五、自动审计记录 .Net 登陆的时候添加验证码 使用Topshelf开发Windows服务、记录日志 日常杂记——C#验证码 c#_生成图片式验证码 C# 利用SharpZipLib生成压缩包 Sql2012如何将远程服务器数据库及表、表结构、表数据导入本地数据库
C# DateTime的11种构造函数 别的也不多说没直接贴代码 using System; using System.Collections.Generic; using System.Glob ...
- C#开发windows服务如何调试——资料整理
原文标题:C# Windows服务程序如何进行调试 原文地址:https://jingyan.baidu.com/article/456c463b18e1b00a583144b3.html 第一种: ...
- 使用Visual Studio 2015 Community 开发windows服务
昨天研究在.NET下开发Windows服务程序,期间遇到一些小问题,这里仅将自己的开发过程和需要注意的地方写下和广大网友分享…… 1.基础 Windows服务是指系统启动时能够自己运行的程序.W ...
- VS2013开发Windows服务项目
这篇随笔里,我将介绍如何用VS2013开发Windows服务项目,实现的功能是定时发送电子邮件. 开发环境:VS2013,SQL Server2008,采用C#语言开发 步骤一:创建Windows服务 ...
- 使用Topshelf开发Windows服务、log4net记录日志
开发windows服务,除了在vs里新建服务项目外(之前有写过具体开发方法,可点击查看),还可以使用Topshelf. 不过使用topshelf需要.netframework 4.5.2版本,在vs2 ...
- 开发Windows服务
在开发Windows服务时需要注意一点,如果在开发完成后,需要通过命令来进行安装的,那么在开发的时候,需要在服务类上面添加一个安装文件.如下图: 添加完成后,就 ...
- c#金额转换成中文大写金额 .Net开发Windows服务
c#金额转换成中文大写金额 2018-08-24 转别人 c#金额转换成中文大写金额 /// <summary> /// 金额转换成中文大写金额 /// </summary> ...
- C#常规开发Windows服务
.Net平台下开发Windows服务的支持库很多,除了通过标准的Windows服务项目,还有一些优秀的开源架构比如:TopSelf:本文以常规项目为例 一.开发 1.新建[Windows服务] 项目: ...
随机推荐
- 重装助手教你如何禁用Windows 10快速启动
快速启动是首先在Windows 8中实现并延续到Windows 10的功能,可在启动PC时提供更快的启动时间.它是一个方便的功能,也是大多数人在不知情的情况下使用的功能,但还有一些功能会在他们掌握新P ...
- Windows程序设计:格式化对话框的设计
刚开始学习Windows程序设计,磕磕碰碰,先做个小笔记缓缓神经,主要是将MessageBox这个Windows API函数的. MessageBox函数是许多人刚开始学习Windows程序设计或者是 ...
- window批处理修改计算机名
一.需要重启 @echo offset /p pcnanme=请输入计算机的名字:reg add HKEY_LOCAL_MACHINE\SYSTEM\ControlSet001\Control\Com ...
- TableExistsException: hbase:namespace
解决:zookeeper还保留着上一次的Hbase设置,所以造成了冲突.删除zookeeper信息,重启之后就没问题了 1.切换到zookeeper的bin目录: 2.执行$sh zkCli.sh 输 ...
- Jetson TX2(1)ubutu1604--安装Nvidia Linux驱动
https://www.jianshu.com/p/c8ebe4aaa708 系统开机首次进入的是以nvidia用户登录的Ubuntu 命令行界面.Nvidia 驱动安装 通过sudo su 输入密码 ...
- Linux内存管理 (3)内核内存的布局图
专题:Linux内存管理专题 关键词:内核内存布局图.lowmem线性映射区.kernel image.ZONE_NORMAL.ZONE_HIGHMEM.swapper_pg_dir.fixmap.v ...
- Java面试准备之数据库
一.考察点 1.联结 1.1 联结的概念: 简单的说,联结是一种机制,用来在一条SELECT语句中关联表,因此称之为联结. 1.2 联结的分类 注意:联结并不代表只有使用join关键字的才算是联结,w ...
- css3学习系列之移动
transform功能 放缩 使用sacle方法实现文字或图像的放缩处理,在参数中指定缩放倍率,比如sacle(0.5)表示缩小50%,例子如下: <!DOCTYPE html> < ...
- Linux命令1
1.获取当前系统支持的所有命令的列表: compgen -c 2.怎样查看一个linux命令的概要与用法: whatis grep #便可查到grep的用法 3.怎样一页一页地查看一个大文件的内 ...
- 使用.net core efcore根据数据库结构自动生成实体类
源码 github,已更新最新代码 https://github.com/leoparddne/GenEntities/ 使用的DB是mysql,所有先nuget一下mysql.data 创建t4模板 ...