用.NET创建Windows服务[转载]

本文介绍了用.NET创建Windows服务的方法,包括什么是Windows服务、创建步骤、服务构成、数据库表脚本样例等。还说明了添加服务安装程序、调试服务及附加正在运行服务的操作,最后总结了创建、安装和调试Windows服务的要点。
 

用.NET创建Windows服务

译者说明:我是通过翻译来学习C#的,文中涉及到的有Visual Studio.NET有关操作,我都根据中文版的VS.NET显示信息来处理的,可以让大家不致有误解。

作者:Mark Strawmyer
 
我们将研究如何创建一个作为Windows服务的应用程序。内容包含什么是Windows服务,如何创建、安装和调试它们。会用到System.ServiceProcess.ServiceBase命名空间的类。


什么是Windows服务?


Windows服务应用程序是一种需要长期运行的应用程序,它对于服务器环境特别适合。它没有用户界面,并且也不会产生任何可视输出。任何用户消息都会被写进Windows事件日志。计算机启动时,服务会自动开始运行。它们不要用户一定登录才运行,它们能在包括这个系统内的任何用户环境下运行。通过服务控制管理器,Windows服务是可控的,可以终止、暂停及当需要时启动。

Windows 服务,以前的NT服务,都是被作为Windows NT操作系统的一部分引进来的。它们在Windows 9x及Windows Me下没有。你需要使用NT级别的操作系统来运行Windows服务,诸如:Windows NT、Windows 2000 Professional或Windows 2000 Server。举例而言,以Windows服务形式的产品有:Microsoft Exchange、SQL Server,还有别的如设置计算机时钟的Windows Time服务。


创建一个Windows服务

我们即将创建的这个服务除了演示什么也不做。服务被启动时会把一个条目信息登记到一个数据库当中来指明这个服务已经启动了。在服务运行期间,它会在指定的时间间隔内定期创建一个数据库项目记录。服务停止时会创建最后一条数据库记录。这个服务会自动向Windows应用程序日志当中登记下它成功启动或停止时的记录。

Visual Studio .NET能够使创建一个Windows服务变成相当简单的一件事情。启动我们的演示服务程序的说明概述如下。

1. 新建一个项目
2. 从一个可用的项目模板列表当中选择Windows服务
3. 设计器会以设计模式打开
4. 从工具箱的组件表当中拖动一个Timer对象到这个设计表面上 (注意: 要确保是从组件列表而不是从Windows窗体列表当中使用Timer)
5. 设置Timer属性,Enabled属性为False,Interval属性30000毫秒
6. 切换到代码视图页(按F7或在视图菜单当中选择代码),然后为这个服务填加功能


Windows服务的构成

在你类后面所包含的代码里,你会注意到你所创建的Windows服务扩充了System.ServiceProcess.Service类。所有以.NET方式建立的Windows服务必须扩充这个类。它会要求你的服务重载下面的方法,Visual Studio默认时包括了这些方法。

• Dispose – 清除任何受控和不受控资源(managed and unmanaged resources)
• OnStart – 控制服务启动
• OnStop – 控制服务停止

数据库表脚本样例

在这个例子中使用的数据库表是使用下面的T-SQL脚本创建的。我选择SQL Server数据库。你可以很容易修改这个例子让它在Access或任何你所选择的别的数据库下运行。

None.gif CREATE   TABLE   [ dbo ] . [ MyServiceLog ]  (
None.gif   
[ in_LogId ]   [ int ]   IDENTITY  ( 1 1 NOT   NULL ,
None.gif   
[ vc_Status ]   [ nvarchar ]  ( 40
None.gif           COLLATE SQL_Latin1_General_CP1_CI_AS 
NOT   NULL ,
None.gif   
[ dt_Created ]   [ datetime ]   NOT   NULL
None.gif
ON   [ PRIMARY ]
None.gif
None.gif

Windows服务样例

下面就是我命名为MyService的Windows服务的所有源代码。大多数源代码是由Visual Studio自动生成的。


None.gif using  System;
None.gif
using  System.Collections;
None.gif
using  System.ComponentModel;
None.gif
using  System.Data;
None.gif
using  System.Data.SqlClient;
None.gif
using  System.Diagnostics;
None.gif
using  System.ServiceProcess;
None.gif
None.gif
namespace  CodeGuru.MyWindowsService
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
InBlock.gif  
public class MyService : System.ServiceProcess.ServiceBase
ExpandedSubBlockStart.gifContractedSubBlock.gif  
dot.gif{
InBlock.gif   
private System.Timers.Timer timer1;
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <remarks> 
InBlock.gif   
/// Required designer variable.
ExpandedSubBlockEnd.gif   
/// </remarks>

InBlock.gif   private System.ComponentModel.Container components = null;
InBlock.gif
InBlock.gif   
public MyService()
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif       
// This call is required by the Windows.Forms 
InBlock.gif       
// Component Designer.
InBlock.gif
     InitializeComponent();
ExpandedSubBlockEnd.gif   }

InBlock.gif
InBlock.gif   
// The main entry point for the process
InBlock.gif
   static void Main()
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     System.ServiceProcess.ServiceBase[] ServicesToRun;
InBlock.gif   
InBlock.gif     ServicesToRun 
= new System.ServiceProcess.ServiceBase[] 
ExpandedSubBlockStart.gifContractedSubBlock.gif
dot.gifnew MyService() };
InBlock.gif
InBlock.gif     System.ServiceProcess.ServiceBase.Run(ServicesToRun);
ExpandedSubBlockEnd.gif   }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <summary> 
InBlock.gif   
/// Required method for Designer support - do not modify 
InBlock.gif   
/// the contents of this method with the code editor.
ExpandedSubBlockEnd.gif   
/// </summary>

InBlock.gif   private void InitializeComponent()
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
this.timer1 = new System.Timers.Timer();
InBlock.gif     ((System.ComponentModel.ISupportInitialize)
InBlock.gif(
this.timer1)).BeginInit();
InBlock.gif     
// 
InBlock.gif     
// timer1
InBlock.gif     
// 
InBlock.gif
     this.timer1.Interval = 30000;
InBlock.gif     
this.timer1.Elapsed += 
InBlock.gif   
new System.Timers.ElapsedEventHandler(this.timer1_Elapsed);
InBlock.gif     
// 
InBlock.gif     
// MyService
InBlock.gif     
// 
InBlock.gif
     this.ServiceName = "My Sample Service";
InBlock.gif     ((System.ComponentModel.ISupportInitialize)
InBlock.gif(
this.timer1)).EndInit();
InBlock.gif
ExpandedSubBlockEnd.gif   }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <summary>
InBlock.gif   
/// Clean up any resources being used.
ExpandedSubBlockEnd.gif   
/// </summary>

InBlock.gif   protected override void Dispose( bool disposing )
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
if( disposing )
ExpandedSubBlockStart.gifContractedSubBlock.gif     
dot.gif{
InBlock.gif      
if (components != null
ExpandedSubBlockStart.gifContractedSubBlock.gif      
dot.gif{
InBlock.gif         components.Dispose();
ExpandedSubBlockEnd.gif      }

ExpandedSubBlockEnd.gif     }

InBlock.gif     
base.Dispose( disposing );
ExpandedSubBlockEnd.gif   }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <summary>
InBlock.gif   
/// Set things in motion so your service can do its work.
ExpandedSubBlockEnd.gif   
/// </summary>

InBlock.gif   protected override void OnStart(string[] args)
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
this.timer1.Enabled = true;
InBlock.gif     
this.LogMessage("Service Started");
ExpandedSubBlockEnd.gif   }

InBlock.gif 
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <summary>
InBlock.gif   
/// Stop this service.
ExpandedSubBlockEnd.gif   
/// </summary>

InBlock.gif   protected override void OnStop()
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
this.timer1.Enabled = false;
InBlock.gif     
this.LogMessage("Service Stopped");
ExpandedSubBlockEnd.gif   }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//*
InBlock.gif    * Respond to the Elapsed event of the timer control
ExpandedSubBlockEnd.gif    
*/

InBlock.gif   
private void timer1_Elapsed(object sender, 
InBlock.gifSystem.Timers.ElapsedEventArgs e)
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
this.LogMessage("Service Running");
ExpandedSubBlockEnd.gif   }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//*
InBlock.gif    * Log specified message to database
ExpandedSubBlockEnd.gif    
*/

InBlock.gif   
private void LogMessage(string Message)
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     SqlConnection connection 
= null;
InBlock.gif     SqlCommand command 
= null;
InBlock.gif     
try
ExpandedSubBlockStart.gifContractedSubBlock.gif     
dot.gif{
InBlock.gif      connection 
= new SqlConnection( 
InBlock.gif
"Server=localhost;Database=SampleDatabase;Integrated 
InBlock.gif
Security=false;User Id=sa;Password=;");
InBlock.gif
command = new SqlCommand(
InBlock.gif
"INSERT INTO MyServiceLog (vc_Status, dt_Created) 
InBlock.gif
VALUES ('" + Message + "',getdate())", connection);
InBlock.gif
      connection.Open();
InBlock.gif      
int numrows = command.ExecuteNonQuery();
ExpandedSubBlockEnd.gif     }

InBlock.gif     
catch( Exception ex )
ExpandedSubBlockStart.gifContractedSubBlock.gif     
dot.gif{
InBlock.gif      System.Diagnostics.Debug.WriteLine(ex.Message);
ExpandedSubBlockEnd.gif     }

InBlock.gif     
finally
ExpandedSubBlockStart.gifContractedSubBlock.gif     
dot.gif{
InBlock.gif      command.Dispose();
InBlock.gif      connection.Dispose();
ExpandedSubBlockEnd.gif     }

ExpandedSubBlockEnd.gif   }

ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}

None.gif
None.gif
None.gif
安装Windows服务

Windows服务不同于普通Windows应用程序。不可能简简单单地通过运行一个EXE就启动Windows服务了。安装一个Windows服务应该通过使用.NET Framework提供的InstallUtil.exe来完成,或者通过诸如一个Microsoft Installer (MSI)这样的文件部署项目完成。


添加服务安装程序

创建一个Windows服务,仅用InstallUtil程序去安装这个服务是不够的。你必须还要把一个服务安装程序添加到你的Windows服务当中,这样便于InstallUtil或是任何别的安装程序知道应用你服务的是怎样的配置设置。

1. 将这个服务程序切换到设计视图
2. 右击设计视图选择“添加安装程序”
3. 切换到刚被添加的ProjectInstaller的设计视图
4. 设置serviceInstaller1组件的属性:
    1) ServiceName = My Sample Service
    2) StartType = Automatic
5. 设置serviceProcessInstaller1组件的属性
    1) Account = LocalSystem
6. 生成解决方案

在完成上面的几个步骤之后,会自动由Visual Studio产生下面的源代码,它包含于ProjectInstaller.cs这个源文件内。

None.gif using  System;
None.gif
using  System.Collections;
None.gif
using  System.ComponentModel;
None.gif
using  System.Configuration.Install;
None.gif
None.gif
namespace  CodeGuru.MyWindowsService
ExpandedBlockStart.gifContractedBlock.gif
dot.gif {
ExpandedSubBlockStart.gifContractedSubBlock.gif  
/**//// <summary>
InBlock.gif  
/// Summary description for ProjectInstaller.
ExpandedSubBlockEnd.gif  
/// </summary>

InBlock.gif  [RunInstaller(true)]
InBlock.gif  
public class ProjectInstaller : 
InBlock.gifSystem.Configuration.Install.Installer
ExpandedSubBlockStart.gifContractedSubBlock.gif  
dot.gif{
InBlock.gif   
private System.ServiceProcess.ServiceProcessInstaller 
InBlock.gifserviceProcessInstaller1;
InBlock.gif   
private System.ServiceProcess.ServiceInstaller serviceInstaller1;
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <summary>
InBlock.gif   
/// Required designer variable.
ExpandedSubBlockEnd.gif   
/// </summary>

InBlock.gif   private System.ComponentModel.Container components = null;
InBlock.gif
InBlock.gif   
public ProjectInstaller()
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
// This call is required by the Designer.
InBlock.gif
     InitializeComponent();
InBlock.gif
InBlock.gif     
// TODO: Add any initialization after the InitComponent call
ExpandedSubBlockEnd.gif
   }

InBlock.gif
ContractedSubBlock.gifExpandedSubBlockStart.gif   
Component Designer generated code#region Component Designer generated code
ExpandedSubBlockStart.gifContractedSubBlock.gif   
/**//// <summary>
InBlock.gif   
/// Required method for Designer support - do not modify
InBlock.gif   
/// the contents of this method with the code editor.
ExpandedSubBlockEnd.gif   
/// </summary>

InBlock.gif   private void InitializeComponent()
ExpandedSubBlockStart.gifContractedSubBlock.gif   
dot.gif{
InBlock.gif     
this.serviceProcessInstaller1 = new 
InBlock.gifSystem.ServiceProcess.ServiceProcessInstaller();
InBlock.gif     
this.serviceInstaller1 = new 
InBlock.gifSystem.ServiceProcess.ServiceInstaller();
InBlock.gif     
// 
InBlock.gif     
// serviceProcessInstaller1
InBlock.gif     
// 
InBlock.gif
     this.serviceProcessInstaller1.Account = 
InBlock.gifSystem.ServiceProcess.ServiceAccount.LocalSystem;
InBlock.gif     
this.serviceProcessInstaller1.Password = null;
InBlock.gif     
this.serviceProcessInstaller1.Username = null;
InBlock.gif     
// 
InBlock.gif     
// serviceInstaller1
InBlock.gif     
// 
InBlock.gif
     this.serviceInstaller1.ServiceName = "My Sample Service";
InBlock.gif     
this.serviceInstaller1.StartType = 
InBlock.gifSystem.ServiceProcess.ServiceStartMode.Automatic;
InBlock.gif     
// 
InBlock.gif     
// ProjectInstaller
InBlock.gif     
// 
InBlock.gif
     this.Installers.AddRange(new 
InBlock.gifSystem.Configuration.Install.Installer[] 
ExpandedSubBlockStart.gifContractedSubBlock.gif
dot.gif{this.serviceProcessInstaller1, this.serviceInstaller1});
ExpandedSubBlockEnd.gif}

ExpandedSubBlockEnd.gif   
#endregion

ExpandedSubBlockEnd.gif  }

ExpandedBlockEnd.gif}

None.gif
None.gif


用InstallUtil安装Windows服务

现在这个服务已经生成,你需要把它安装好才能使用。下面操作会指导你安装你的新服务。

1. 打开Visual Studio .NET命令提示
2. 改变路径到你项目所在的bin\Debug文件夹位置(如果你以Release模式编译则在bin\Release文件夹)
3. 执行命令“InstallUtil.exe MyWindowsService.exe”注册这个服务,使它建立一个合适的注册项。
4. 右击桌面上“我的电脑”,选择“管理”就可以打计算机管理控制台
5. 在“服务和应用程序”里面的“服务”部分里,你可以发现你的Windows服务已经包含在服务列表当中了
6. 右击你的服务选择启动就可以启动你的服务了

在每次需要修改Windows服务时,这就会要求你卸载和重新安装这个服务。不过要注意在卸载这个服务前,最好确保服务管理控制台已经关闭,这会是一个很好的习惯。如果没有这样操作的话,你可能在卸载和重安装Windows服务时会遇到麻烦。仅卸载服务的话,可以执行相的InstallUtil命令用于注销服务,不过要在后面加一个/u命令开关。


调试Windows服务

从另外的角度度看,调试Windows服务绝不同于一个普通的应用程序。调试Windows服务要求的步骤更多。服务不能象你对普通应用程序做的那样,只要简单地在开发环境下执行就可以调试了。服务必须首先被安装和启动,这一点在前面部分我们已经做到了。为了便于跟踪调试代码,一旦服务被启动,你就要用Visual Studio把运行的进程附加进来(attach)。记住,对你的Windows服务做的任何修改都要对这个服务进行卸载和重安装。


附加正在运行的Windows服务

为了调试程序,有些附加Windows服务的操作说明。这些操作假定你已经安装了这个Windows服务并且它正在运行。

1. 用Visual Studio装载这个项目
2. 点击“调试”菜单
3. 点击“进程”菜单
4. 确保 显示系统进程 被选
5. 在 可用进程 列表中,把进程定位于你的可执行文件名称上点击选中它
6. 点击 附加 按钮
7. 点击 确定
8. 点击 关闭
9. 在timer1_Elapsed方法里设置一个断点,然后等它执行


总结

现在你应该对Windows服务是什么,以及如何创建、安装和调试它们有一个粗略的认识了。Windows服务的额处的功能你可以自行研究。这些功能包括暂停(OnPause)和恢复(OnContinue)的能力。暂停和恢复的能力在默认情况下没有被启用,要通过Windows服务属性来设置。


About the Author
Mark Strawmyer, MCSD, MCSE (NT4/W2K), MCDBA is a Senior Architect of .NET applications for large and mid-size organizations. Mark is a technology leader with Crowe Chizek in Indianapolis, Indiana. He specializes in architecture, design and development of Microsoft-based solutions. You can reach Mark at mstrawmyer@crowechizek.com.


翻译
作者Blog: http://blog.youkuaiyun.com/ego/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值