用c语言编写windows服务程序

上传人:子 文档编号:41627795 上传时间:2018-05-30 格式:DOC 页数:6 大小:33.50KB
返回 下载 相关 举报
用c语言编写windows服务程序_第1页
第1页 / 共6页
用c语言编写windows服务程序_第2页
第2页 / 共6页
用c语言编写windows服务程序_第3页
第3页 / 共6页
用c语言编写windows服务程序_第4页
第4页 / 共6页
用c语言编写windows服务程序_第5页
第5页 / 共6页
点击查看更多>>
资源描述

《用c语言编写windows服务程序》由会员分享,可在线阅读,更多相关《用c语言编写windows服务程序(6页珍藏版)》请在金锄头文库上搜索。

1、用用 C 语言编写语言编写 Windows 服务程序的五个步骤服务程序的五个步骤原文出处:Five Steps to Writing Windows Services in C摘要摘要Windows 服务被设计用于需要在后台运行的应用程序以及实现没有用户交互的任务。 为了学习这种控制台应用程序的基础知识,C(不是 C+)是最佳选择。本文将建立并 实 现一个简单的服务程序,其功能是查询系统中可用物理内存数量,然后将结果写入一个文 本文件。最后,你可以用所学知识编写自己的 Windows 服务。当初我写第一个 NT 服务时,我到 MSDN 上找例子。在那里我找到了一篇 Nigel Thompson

2、 写的文章:“Creating a Simple Win32 Service in C+”,这篇文章附带一个 C+ 例子。虽然这篇文章很好地解释了服务的开发过程,但是,我仍然感觉缺少我需要的重要 信息。我想理解通过什么框架,调用什么函数,以及何时调用,但 C+ 在这方面没有让 我轻松多少。面向对象的方法固然方便,但由于用类对底层 Win32 函数调用进行了封装, 它不利于学习服务程序的基本知识。这就是为什么我觉得 C 更加适合于编写初级服务程 序或者实现简单后台任务的服务。在你对服务程序有了充分透彻的理解之后,用 C+ 编 写才能游刃有余。当我离开原来的工作岗位,不得不向另一个人转移我的知识的

3、时候,利 用我用 C 所写的例子就非常容易解释 NT 服务之所以然。服务是一个运行在后台并实现勿需用户交互的任务的控制台程序。Windows NT/2000/XP 操作系统提供为服务程序提供专门的支持。人们可以用服务控制面板来配置 安装好的服务程序,也就是 Windows 2000/XP 控制面板|管理工具中的“服务”(或在“开始” |“运行”对话框中输入 services.msc /s译者注) 。可以将服务配置成操作系统启动时自动 启动,这样你就不必每次再重启系统后还要手动启动服务。本文将首先解释如何创建一个定期查询可用物理内存并将结果写入某个文本文件的服 务。然后指导你完成生成,安装和实现

4、服务的整个过程。第一步:主函数和全局定义第一步:主函数和全局定义首先,包含所需的头文件。例子要调用 Win32 函数(windows.h)和磁盘文件写入 (stdio.h): #include #include 接着,定义两个常量: #define SLEEP_TIME 5000 #define LOGFILE “C:MyServicesmemstatus.txt“ SLEEP_TIME 指定两次连续查询可用内存之间的毫秒间隔。在第二步中编写服务工作循环 的时候要使用该常量。LOGFILE 定义日志文件的路径,你将会用 WriteToLog 函数将内存查询的结果输出到该 文件,WriteToL

5、og 函数定义如下: int WriteToLog(char* str) FILE* log;log = fopen(LOGFILE, “a+“);if (log = NULL)return -1;fprintf(log, “%sn“, str);fclose(log);return 0; 声明几个全局变量,以便在程序的多个函数之间共享它们值。此外,做一个函数的前向定 义: SERVICE_STATUS ServiceStatus; SERVICE_STATUS_HANDLE hStatus; void ServiceMain(int argc, char* argv); void Contr

6、olHandler(DWORD request); int InitService();现在,准备工作已经就绪,你可以开始编码了。服务程序控制台程序的一个子集。因 此,开始你可以定义一个 main 函数,它是程序的入口点。对于服务程序来说,main 的代 码令人惊讶地简短,因为它只创建分派表并启动控制分派机。 void main() SERVICE_TABLE_ENTRY ServiceTable2;ServiceTable0.lpServiceName = “MemoryStatus“;ServiceTable0.lpServiceProc = (LPSERVICE_MAIN_FUNCTIO

7、N)ServiceMain;ServiceTable1.lpServiceName = NULL;ServiceTable1.lpServiceProc = NULL;/ 启动服务的控制分派机线程StartServiceCtrlDispatcher(ServiceTable); 一个程序可能包含若干个服务。每一个服务都必须列于专门的分派表中(为此该程序 定义了一个 ServiceTable 结构数组) 。这个表中的每一项都要在 SERVICE_TABLE_ENTRY 结构之中。它有两个域: lpServiceName: 指向表示服务名称字符串的指针;当定义了多个服务时,那么这个 域必须指定;

8、lpServiceProc: 指向服务主函数的指针(服务入口点) ;分派表的最后一项必须是服务名和服务主函数域的 NULL 指针,文本例子程序中只 宿主一个服务,所以服务名的定义是可选的。服务控制管理器(SCM:Services Control Manager)是一个管理系统所有服务的进程。 当 SCM 启动某个服务时,它等待某个进程的主线程来调用 StartServiceCtrlDispatcher 函 数。将分派表传递给 StartServiceCtrlDispatcher。这将把调用进程的主线程转换为控制分派 器。该分派器启动一个新线程,该线程运行分派表中每个服务 的 ServiceMa

9、in 函数(本 文例子中只有一个服务)分派器还监视程序中所有服务的执行情况。然后分派器将控制请 求从 SCM 传给服务。注意:如果 StartServiceCtrlDispatcher 函数30秒没有被调用,便会报错,为了避免这种情 况,我们必须在 ServiceMain 函数中(参见本文例子)或在非主函数的单独线程中初始化 服务分派表。本文所描述的服务不需要防范这样的情况。分派表中所有的服务执行完之后(例如,用户通过“服务”控制面板程序停止它们) , 或者发生错误时。StartServiceCtrlDispatcher 调用返回。然后主进程终止。第二步:第二步:ServiceMain 函数函

10、数Listing 1 展示了 ServiceMain 的代码。该函数是服务的入口点。它运行在一个单独 的线程当中,这个线程是由控制分派器创建的。ServiceMain 应该尽可能早早为服务注册 控制处理器。这要通过调用 RegisterServiceCtrlHadler 函数来实现。你要将两个参数传递 给此函数:服务名和指向 ControlHandlerfunction 的指针。它指示控制分派器调用 ControlHandler 函数处理 SCM 控制请求。注册完控制处理 器之后,获得状态句柄(hStatus) 。通过调用 SetServiceStatus 函数,用 hStatus 向 SCM

11、 报告服务的状态。Listing 1 展示了如何指定服务特征和其当前状态来初始化 ServiceStatus 结构, ServiceStatus 结构的每个域都有其用途: dwServiceType:指示服务类型,创建 Win32 服务。赋值 SERVICE_WIN32; dwCurrentState:指定服务的当前状态。因为服务的初始化在这里没有完成,所以 这里的状态为 SERVICE_START_PENDING; dwControlsAccepted:这个域通知 SCM 服务接受哪个域。本文例子是允许 STOP 和 SHUTDOWN 请求。处理控制请求将在第三步讨论; dwWin32Exi

12、tCode 和 dwServiceSpecificExitCode:这两个域在你终止服务并报告退 出细节时很有用。初始化服务时并不退出,因此,它们的值为 0; dwCheckPoint 和 dwWaitHint:这两个域表示初始化某个服务进程时要30秒以上。 本文例子服务的初始化过程很短,所以这两个域的值都为 0。调用 SetServiceStatus 函数向 SCM 报告服务的状态时。要提供 hStatus 句柄和 ServiceStatus 结构。注意 ServiceStatus 一个全局变量,所以你可以跨多个函数使用它。 ServiceMain 函数中,你给结构的几个域赋值,它们在服务运

13、行的整个过程中都保持不变, 比如:dwServiceType。在报告了服务状态之后,你可以调用 InitService 函数来完成初始化。这个函数只是 添加一个说明性字符串到日志文件。如下面代码所示: / 服务初始化 int InitService() int result;result = WriteToLog(“Monitoring started.“);return(result); 在 ServiceMain 中,检查 InitService 函数的返回值。如果初始化有错(因为有可能写 日志文件失败) ,则将服务状态置为终止并退出 ServiceMain:error = InitSer

14、vice(); if (error) / 初始化失败,终止服务ServiceStatus.dwCurrentState = SERVICE_STOPPED; ServiceStatus.dwWin32ExitCode = -1; SetServiceStatus(hStatus, / 退出 ServiceMainreturn; 如果初始化成功,则向 SCM 报告状态: / 向 SCM 报告运行状态 ServiceStatus.dwCurrentState = SERVICE_RUNNING; SetServiceStatus (hStatus, 接着,启动工作循环。每五秒钟查询一个可用物理内存

15、并将结果写入日志文件。如 Listing 1 所示,循环一直到服务的状态为 SERVICE_RUNNING 或日志文件写入出错 为止。状态可能在 ControlHandler 函数响应 SCM 控制请求时修改。第三步:处理控制请求第三步:处理控制请求在第二步中,你用 ServiceMain 函数注册了控制处理器函数。控制处理器与处理各 种 Windows 消息的窗口回调函数非常类似。它检查 SCM 发送了什么请求并采取相应行 动。每次你调用 SetServiceStatus 函数的时候,必须指定服务接收 STOP 和 SHUTDOWN 请求。Listing 2 示范了如何在 ControlHa

16、ndler 函数中处理它们。STOP 请求是 SCM 终止服务的时候发送的。例如,如果用户在“服务”控制面板中手 动终止服务。SHUTDOWN 请求是关闭机器时,由 SCM 发送给所有运行中服务的请求。 两种情况的处理方式相同: 写日志文件,监视停止; 向 SCM 报告 SERVICE_STOPPED 状态;由于 ServiceStatus 结构对于整个程序而言为全局量,ServiceStatus 中的工作循环在当 前状态改变或服务终止后停止。其它的控制请求如:PAUSE 和 CONTINUE 在本文的例 子没有处理。控制处理器函数必须报告服务状态,即便 SCM 每次发送控制请求的时候状态保持 相同。因此,不管响应什么请求,都要调用 SetServiceStatus。图一 显示 MemoryStatus 服务的服务控制面板第四步:安装和配置服务第四步:安装和配置服务程序编好了,将之编译成 exe 文件。本文例子创建的文件叫 MemorySt

展开阅读全文
相关资源
正为您匹配相似的精品文档
相关搜索

最新文档


当前位置:首页 > 生活休闲 > 科普知识

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号