一、Win2000服务简介
服务程序(Service Application)是一种运行于WinNT的后台程序,每个服务 程序(Service Application)中可能包含若干个服务(Service),每个服务就 是其中的一个线程(该服务也可以创建多个子线程)。采用服务,应用程序可以 获得特殊的权限,而且不会被用户通过Win2000的任务管理器直接结束程序,所以 服务常常用来实现一些特殊的目标。
通过Win2000控制面板中的服务管理工具,我们可以设置/查看服务的特性:
(1)服务名称;(2)显示名称;(3)描述;(4)启动类型;(5)依赖关 系;
其中,服务名称是标识给服务的。
以Win2000的C:\WINNT\System32\services.exe程序为例子,该Exe文件对应一个 Service Application,是该服务程序的可见实体;该exe中包含多个服务 (Service),例如Alerter,Dhcp(DHCP Client),Messenger等。当我们结束 一个服务的时候,该服务所在的Service Application中的其他服务并没有被终止 。
在Delphi中,Borland的工程师为我们提供了TServiceApplication,TService ,TServiceThread等类,封装了大量细节,简化了服务程序的开发。
二、TServiceApplication
在Delphi中,类TServiceApplication就对应上述的ServiceApplication。利 用Delphi的开发环境,我们新建一个Service Application Project,同时就创建 了一个继承自TService的类。项目文件中的Application对象就是一个 TServiceApplication实例。每个TServiceApplication包含若干个TService对象 ,正好对应上述的服务程序和服务之间的数量关系。
通过阅读TServiceApplication和TService类的定义,可以得知, TServiceApplication从TComponent类继承而来,TService从类TDataModule基础 而来,Application对象负责各个TService对象的Create和Destroy。跟踪下列代 码
Application.CreateForm(TService1, Service1);
可以发现创建的TService对象的Owner都是Application对象;在VCL Framework中Owner总是负责Destroy各个Component对象(VCL的TComponent类采用 了Composite模式),所以TServiceApplication也将Destroy各个TService对象。
下面跟踪TServiceApplication.Run的代码,可以发现TServiceApplication首 先解析运行参数,实现了服务的Install和Uninstall。然后,初始化一个 ServiceStartTable数组,该数组包含了各个service对象的服务名称和运行入口 ;最后创建一个TServiceStartThread 对象,该对象是一个线程对象,从线程调 用API:StartServiceCtrlDispatcher来启动ServiceStartTable中指定的若干个 服务;而ServiceApplication主线程就不断循环,处理消息,比如接收请求来停 止/暂停某个服务。
三、TService
TService类继承自类 TDataModule,这意味着我们可以加入大量的VCL控件, 实现丰富的功能。此外,我们还可以处理OnStart,OnPause,OnStop, OnContinue,OnCreate,OnShutDown等事件。其中需要说明的是:OnStop表示该 服务被停止;而OnShutDown表示该ServiceApplication停止运行,这意味着其他 服务也被终止了;两者含义是不一样的。
前面讲过,ServiceApplication通过调用StartServiceCtrlDispatcher来启动 各个服务。StartServiceCtrlDispatcher启动TService的入口,该入库就是 TService.Main。TService.Main首先注册该服务,然后调用TService.DoStart。 TService.DoStart创建一个内部TServiceThread成员对象,这是一个线程对象; 考察TServiceThread.Execute可以得知,当我们处理的TService1. OnExecute, 那么TService会把所有的请求委托给该TServiceThread成员对象处理,该对象以 默认的方式处理所有的请求。
TService. ServiceExecute是TService的主体内容。一个服务要正常运行,除 了需要处理它要关注的目标(比如监听某个端口、执行某个任务等)外,还要响 应外部命令/请求:比如终止、暂停、恢复该服务。因此可以考虑创建一个专门的 线程来完成该任务,而在ServiceExecute中处理外面命令/请求。因此代码如下:
while not Terminated do begin
ServiceThread.ProcessRequests(False);
end;
当然,也可以在OnExecute中处理某些任务,如监听某个端口,但是这常常会 导致该Service不能及时响应Stop/Pause等请求。当OnExecute执行完了,该服务 实际上就完成了任务要结束了(terminate)。