下載本文檔
版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認領(lǐng)
文檔簡介
1、用 C 語言編寫 Windows 服務(wù)程序的五個步驟原文出處: Five Steps to Writing Windows Services in C摘要Windows 服務(wù)被設(shè)計用于需要在后臺運行的應用程序以及實現(xiàn)沒有用戶交互的任務(wù)。為了學習這種控制臺應用程序的基礎(chǔ)知識,C(不是C+ )是最佳選擇。本文將建立并實現(xiàn)一個簡單的服務(wù)程序,其功能是查詢系統(tǒng)中可用物理內(nèi)存數(shù)量,然后將結(jié)果寫入一個文本文件。最后,你可以用所學知識編寫自己的Windows 服務(wù)。當初我寫第一個NT 服務(wù)時,我到MSDN 上找例子。在那里我找到了一篇NigelThompson 寫的文章: “Creating a Simpl
2、e Win32 Service in C+ ”,這篇文章附帶一個C+例子。雖然這篇文章很好地解釋了服務(wù)的開發(fā)過程,但是,我仍然感覺缺少我需要的重要信息。我想理解通過什么框架,調(diào)用什么函數(shù),以及何時調(diào)用,但C+在這方面沒有讓我輕松多少。面向?qū)ο蟮姆椒ü倘环奖?,但由于用類對底層Win32 函數(shù)調(diào)用進行了封裝,它不利于學習服務(wù)程序的基本知識。這就是為什么我覺得C 更加適合于編寫初級服務(wù)程序或者實現(xiàn)簡單后臺任務(wù)的服務(wù)。在你對服務(wù)程序有了充分透徹的理解之后,用 C+ 編寫才能游刃有余。當我離開原來的工作崗位,不得不向另一個人轉(zhuǎn)移我的知識的時候,利用我用C 所寫的例子就非常容易解釋NT 服務(wù)之所以然。服
3、務(wù) 是 一 個 運 行 在 后 臺 并 實現(xiàn) 勿 需 用 戶 交 互 的 任 務(wù) 的 控 制 臺 程 序 。 WindowsNT/2000/XP 操作系統(tǒng)提供為服務(wù)程序提供專門的支持。人們可以用服務(wù)控制面板來配置安裝好的服務(wù)程序, 也就是 Windows 2000/XP控制面板 |管理工具中的 “服務(wù) ”(或在 “開始 ”|運“行”對話框中輸入 services.msc /s 譯者注)??梢詫⒎?wù)配置成操作系統(tǒng)啟動時自動啟動,這樣你就不必每次再重啟系統(tǒng)后還要手動啟動服務(wù)。本文將首先解釋如何創(chuàng)建一個定期查詢可用物理內(nèi)存并將結(jié)果寫入某個文本文件的服務(wù)。然后指導你完成生成,安裝和實現(xiàn)服務(wù)的整個過程。
4、第一步:主函數(shù)和全局定義首先,包含所需的頭文件。例子要調(diào)用Win32函數(shù)( windows.h )和磁盤文件寫入 ( stdio.h ):#include <windows.h>#include <stdio.h>接著,定義兩個常量:#define SLEEP_TIME 5000#define LOGFILE "C:MyServicesmemstatus.txt"SLEEP_TIME 指定兩次連續(xù)查詢可用內(nèi)存之間的毫秒間隔。在第二步中編寫服務(wù)工作循環(huán)的時候要使用該常量。LOGFILE定義日志文件的路徑,你將會用文件, WriteToLog函數(shù)定義如下
5、:WriteToLog函數(shù)將內(nèi)存查詢的結(jié)果輸出到該int WriteToLog(char* str)FILE* log;log = fopen(LOGFILE, "a+");if (log = NULL)return -1;fprintf(log, "%sn", str);fclose(log);return 0;聲明幾個全局變量, 以便在程序的多個函數(shù)之間共享它們值。此外,做一個函數(shù)的前向定義:SERVICE_STATUS ServiceStatus;SERVICE_STATUS_HANDLE hStatus;void ServiceMain(int
6、argc, char* argv);void ControlHandler(DWORD request);int InitService();現(xiàn)在,準備工作已經(jīng)就緒,你可以開始編碼了。服務(wù)程序控制臺程序的一個子集。因此,開始你可以定義一個 main 函數(shù),它是程序的入口點。對于服務(wù)程序來說, main 的代碼令人驚訝地簡短,因為它只創(chuàng)建分派表并啟動控制分派機。void main()SERVICE_TABLE_ENTRY ServiceTable2; ServiceTable0.lpServiceName = "MemoryStatus" ServiceTable0.lpSe
7、rviceProc = (LPSERVICE_MAIN_FUNCTION)ServiceMain;ServiceTable1.lpServiceName = NULL;ServiceTable1.lpServiceProc = NULL;/ 啟動服務(wù)的控制分派機線程StartServiceCtrlDispatcher(ServiceTable);一個程序可能包含若干個服務(wù)。每一個服務(wù)都必須列于專門的分派表中(為此該程序定義了一個ServiceTable 結(jié)構(gòu)數(shù)組) 。這個表中的每一項都要在SERVICE_TABLE_ENTRY結(jié)構(gòu)之中。它有兩個域:lpServiceName: 指向表示服務(wù)名稱
8、字符串的指針;當定義了多個服務(wù)時,那么這個域必須指定;lpServiceProc: 指向服務(wù)主函數(shù)的指針(服務(wù)入口點);分派表的最后一項必須是服務(wù)名和服務(wù)主函數(shù)域的NULL指針,文本例子程序中只宿主一個服務(wù),所以服務(wù)名的定義是可選的。服務(wù)控制管理器(SCM : Services Control Manager )是一個管理系統(tǒng)所有服務(wù)的進程。當 SCM啟動某個服務(wù)時,它等待某個進程的主線程來調(diào)用StartServiceCtrlDispatcher函數(shù)。將分派表傳遞給StartServiceCtrlDispatcher 。這將把調(diào)用進程的主線程轉(zhuǎn)換為控制分派器。該分派器啟動一個新線程,該線程運行
9、分派表中每個服務(wù)的 ServiceMain函數(shù)(本文例子中只有一個服務(wù))分派器還監(jiān)視程序中所有服務(wù)的執(zhí)行情況。然后分派器將控制請求從SCM 傳給服務(wù)。注意:如果StartServiceCtrlDispatcher函數(shù) 30秒沒有被調(diào)用,便會報錯,為了避免這種情況,我們必須在 ServiceMain 函數(shù)中(參見本文例子)或在非主函數(shù)的單獨線程中初始化服務(wù)分派表。本文所描述的服務(wù)不需要防范這樣的情況。分派表中所有的服務(wù)執(zhí)行完之后 (例如,用戶通過 “服務(wù) ”控制面板程序停止它們) ,或者發(fā)生錯誤時。 StartServiceCtrlDispatcher 調(diào)用返回。然后主進程終止。第二步:Serv
10、iceMain函數(shù)Listing 1展示了ServiceMain的代碼。該函數(shù)是服務(wù)的入口點。它運行在一個單獨的線程當中,這個線程是由控制分派器創(chuàng)建的。ServiceMain應該盡可能早早為服務(wù)注冊控制處理器。這要通過調(diào)用RegisterServiceCtrlHadler函數(shù)來實現(xiàn)。你要將兩個參數(shù)傳遞給此函數(shù):服務(wù)名和指向ControlHandlerfunction的指針。它指示控制分派器調(diào)用ControlHandler函數(shù)處理SCM 控制請求。 注冊完控制處理器之后,獲得狀態(tài)句柄(hStatus)。通過調(diào)用SetServiceStatus 函數(shù),用hStatus 向SCM 報告服務(wù)的狀態(tài)。L
11、isting1展 示 了 如 何 指 定 服 務(wù) 特 征 和 其 當 前 狀 態(tài) 來 初 始 化ServiceStatus 結(jié) 構(gòu) ,ServiceStatus 結(jié)構(gòu)的每個域都有其用途:dwServiceType:指示服務(wù)類型,創(chuàng)建Win32服務(wù)。賦值SERVICE_WIN32 ;dwCurrentState:指定服務(wù)的當前狀態(tài)。因為服務(wù)的初始化在這里沒有完成,所以這里的狀態(tài)為SERVICE_START_PENDING ;dwControlsAccepted:這個域通知SCM服務(wù)接受哪個域。 本文例子是允許STOP和SHUTDOWN請求。處理控制請求將在第三步討論;dwWin32ExitCod
12、e和 dwServiceSpecificExitCode :這兩個域在你終止服務(wù)并報告退出細節(jié)時很有用。初始化服務(wù)時并不退出,因此,它們的值為0;dwCheckPoint和 dwWaitHint :這兩個域表示初始化某個服務(wù)進程時要本文例子服務(wù)的初始化過程很短,所以這兩個域的值都為0。調(diào)用SetServiceStatus 函數(shù)向SCM報告服務(wù)的狀態(tài)時。要提供30秒以上。hStatus 句柄和ServiceStatus 結(jié)構(gòu)。注意 ServiceStatus 一個全局變量,所以你可以跨多個函數(shù)使用它。 ServiceMain 函數(shù)中,你給結(jié)構(gòu)的幾個域賦值,它們在服務(wù)運行的整個過程中都保持不變,比
13、如: dwServiceType。在報告了服務(wù)狀態(tài)之后, 你可以調(diào)用 InitService 函數(shù)來完成初始化。 這個函數(shù)只是添加一個說明性字符串到日志文件。如下面代碼所示:/ 服務(wù)初始化int InitService()int result;result = WriteToLog("Monitoring started.");return(result);在 ServiceMain 中,檢查 InitService 函數(shù)的返回值。如果初始化有錯(因為有可能寫日志文件失?。?,則將服務(wù)狀態(tài)置為終止并退出 ServiceMain :error = InitService();
14、if (error)/ 初始化失敗,終止服務(wù)ServiceStatus.dwCurrentState = SERVICE_STOPPED;ServiceStatus.dwWin32ExitCode = -1;SetServiceStatus(hStatus, &ServiceStatus);/ 退出 ServiceMain return;如果初始化成功,則向SCM 報告狀態(tài):/ 向 SCM 報告運行狀態(tài)ServiceStatus.dwCurrentState = SERVICE_RUNNING;SetServiceStatus (hStatus, &ServiceStatus)
15、;接著,啟動工作循環(huán)。每五秒鐘查詢一個可用物理內(nèi)存并將結(jié)果寫入日志文件。如 Listing 1 所示,循環(huán)一直到服務(wù)的狀態(tài)為為止。狀態(tài)可能在 ControlHandler 函數(shù)響應SERVICE_RUNNINGSCM 控制請求時修改?;蛉罩疚募懭氤鲥e第三步:處理控制請求在第二步中,你用ServiceMain函數(shù)注冊了控制處理器函數(shù)??刂铺幚砥髋c處理各種Windows 消息的窗口回調(diào)函數(shù)非常類似。它檢查SCM 發(fā)送了什么請求并采取相應行動。每次你調(diào)用SetServiceStatus 函數(shù)的時候, 必須指定服務(wù)接收STOP 和 SHUTDOWN請求。 Listing 2示范了如何在Control
16、Handler函數(shù)中處理它們。STOP 請求是SCM 終止服務(wù)的時候發(fā)送的。例如,如果用戶在“服務(wù) ”控制面板中手動終止服務(wù)。SHUTDOWN請求是關(guān)閉機器時,由SCM發(fā)送給所有運行中服務(wù)的請求。兩種情況的處理方式相同:寫日志文件,監(jiān)視停止;向 SCM 報告 SERVICE_STOPPED 狀態(tài);由于ServiceStatus 結(jié)構(gòu)對于整個程序而言為全局量,ServiceStatus 中的工作循環(huán)在當前狀態(tài)改變或服務(wù)終止后停止。其它的控制請求如:PAUSE 和 CONTINUE在本文的例子沒有處理??刂铺幚砥骱瘮?shù)必須報告服務(wù)狀態(tài),即便SCM每次發(fā)送控制請求的時候狀態(tài)保持相同。因此,不管響應什么
17、請求,都要調(diào)用SetServiceStatus。圖一顯示MemoryStatus 服務(wù)的服務(wù)控制面板第四步:安裝和配置服務(wù)程序編好了, 將之編譯成 exe 文件。本文例子創(chuàng)建的文件叫 MemoryStatus.exe,將它拷貝到 C:MyServices 文件夾。 為了在機器上安裝這個服務(wù), 需要用 SC.EXE 可執(zhí)行文件,它是 Win32 Platform SDK 中附帶的一個工具。 (譯者注: Visaul Studio .NET 2003 IDE 環(huán)境中 也 有 這 個 工 具 , 具 體 存 放 位 置 在 : C:Program FilesMicrosoft Visual Stud
18、io .NET 2003Common7ToolsBinwinnt )。使用這個實用工具可以安裝和移除服務(wù)。其它控制操作將通過服務(wù)控制面板來完成。以下是用命 令行安裝 MemoryStatus 服務(wù)的方法:sc create MemoryStatus binpath= c:MyServicesMemoryStatus.exe發(fā)出此創(chuàng)建命令。指定服務(wù)名和二進制文件的路徑(注意 binpath= 空格)。安裝成功后, 便可以用服務(wù)控制面板來控制這個服務(wù)(參見圖一)具欄啟動和終止這個服務(wù)。和路徑之間的那個。用控制面板的工圖二MemoryStatus 服務(wù)的屬性窗口MemoryStatus 的啟動類型是
19、手動, 也就是說根據(jù)需要來啟動這個服務(wù)。 右鍵單擊該服務(wù),然后選擇上下文菜單中的 “屬性 ”菜單項, 此時顯示該服務(wù)的屬性窗口。 在這里可以修改啟動類型以及其它設(shè)置。你還可以從 “常規(guī) ”標簽中啟動 /停止服務(wù)。以下是從系統(tǒng)中移除服務(wù)的方法:sc delete MemoryStatus指定 “delete ”選項和服務(wù)名。此服務(wù)將被標記為刪除,下次西通重啟后,該服務(wù)將被完全移除。第五步:測試服務(wù)從服務(wù)控制面板啟動MemoryStatus 服務(wù)。如果初始化不出錯,表示啟動成功。過一會兒將服務(wù)停止。檢查一下 C:MyServices 文件夾中 memstatus.txt 文件的服務(wù)輸出。在我的機器上輸出是這樣的:Monitoring started.273469440273379出師表兩漢:諸葛亮先帝創(chuàng)業(yè)未半而中道崩殂, 今天下三分, 益州疲弊, 此誠危急存亡之秋也。然侍衛(wèi)之臣不懈于內(nèi),忠志之士忘身于外者,蓋追先帝之殊遇,欲報之于陛下也。誠宜開張圣聽,以光先帝遺德,恢弘志士之氣,不宜妄自菲薄,引喻失義,以塞忠
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預覽,若沒有圖紙預覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負責。
- 6. 下載文件中如有侵權(quán)或不適當內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準確性、安全性和完整性, 同時也不承擔用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 太陽能儲能系統(tǒng)-深度研究
- 2025年廣東職業(yè)技術(shù)學院高職單招職業(yè)技能測試近5年??及鎱⒖碱}庫含答案解析
- 3D建模與重構(gòu)技術(shù)-深度研究
- 2025年山東鋁業(yè)職業(yè)學院高職單招高職單招英語2016-2024歷年頻考點試題含答案解析
- 冬季校園除雪施工方案
- 基于社區(qū)老年生活圈的開封龍亭區(qū)“體醫(yī)養(yǎng)”公共服務(wù)設(shè)施配置與優(yōu)化
- 2025年安徽警官職業(yè)學院高職單招語文2018-2024歷年參考題庫頻考點含答案解析
- 2025年安徽工商職業(yè)學院高職單招高職單招英語2016-2024歷年頻考點試題含答案解析
- 混凝土路面修復施工方案
- 2025年太湖創(chuàng)意職業(yè)技術(shù)學院高職單招職業(yè)技能測試近5年??及鎱⒖碱}庫含答案解析
- 2024生態(tài)環(huán)境相關(guān)法律法規(guī)考試試題
- 有砟軌道施工工藝課件
- 兩辦意見八硬措施煤礦安全生產(chǎn)條例宣貫學習課件
- 40篇短文搞定高中英語3500單詞
- 人教版高中數(shù)學必修二《第九章 統(tǒng)計》同步練習及答案解析
- 兒科護理安全警示教育課件
- 三年級下冊口算天天100題
- 國家中英文名稱及代碼縮寫(三位)
- 人員密集場所消防安全培訓
- 液晶高壓芯片去保護方法
- 拜太歲科儀文檔
評論
0/150
提交評論