亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C編寫Windows服務程序的五個步驟

2019-11-17 05:05:45
字體:
來源:轉載
供稿:網友
Windows 服務被設計用于需要在后臺運行的應用程序以及實現沒有用戶交互的任務。為了學習這種控制臺應用程序的基礎知識,C(不是C++)是最佳選擇。本文將建立并實現一個簡單的服務程序,其功能是查詢系統中可用物理內存數量,然后將結果寫入一個文本文件。
最后,你可以用所學知識編寫自己的 Windows 服務。

  當初我寫第一個NT 服務時,我到 MSDN 上找例子。在那里我找到了一篇 Nigel Thompson 寫的文章:“Creating a Simple Win32 Service in C++”,這篇文章附帶一個 C++ 例子。雖然這篇文章很好地解釋了服務的開發過程,但是,我仍然感覺缺少我需要的重要信息。我想理解通過什么框架,調用什么函數,以及何時調用,但 C++ 在這方面沒有讓我輕松多少。面向對象的方法固然方便,但由于用類對底層 Win32 函數調用進行了封裝,它不利于學習服務程序的基本知識。這就是為什么我覺得 C 更加適合于編寫初級服務程序或者實現簡單后臺任務的服務。在你對服務程序有了充分透徹的理解之后,用 C++ 編寫才能游刃有余。當我離開原來的工作崗位,不得不向另一個人轉移我的知識的時候,利用我用 C 所寫的例子就非常輕易解釋 NT 服務之所以然。

  服務是一個運行在后臺并實現勿需用戶交互的任務的控制臺程序。Windows NT/2000/XP 操作系統提供為服務程序提供專門的支持。人們可以用服務控制面板來配置安裝好的服務程序,也就是 windows 2000/XP 控制面板 治理工具中的“服務”(或在“開始” “運行”對話框中輸入 services.msc /s——譯者注)。可以將服務配置成操作系統啟動時自動啟動,這樣你就不必每次再重啟系統后還要手動啟動服務。

  本文將首先解釋如何創建一個定期查詢可用物理內存并將結果寫入某個文本文件的服務。然后指導你完成生成,安裝和實現服務的整個過程。

  第一步:主函數和全局定義

  首先,包含所需的頭文件。例子要調用 Win32 函數(windows.h)和磁盤文件寫入(stdio.h):

  

  #include

  #include

  接著,定義兩個常量:

  

  #define SLEEP_TIME 5000

  #define LOGFILE "C://MyServices//memstatus.txt"

  SLEEP_TIME 指定兩次連續查詢可用內存之間的毫秒間隔。在第二步中編寫服務工作循環的時候要使用該常量。

  LOGFILE 定義日志文件的路徑,你將會用 WriteToLog 函數將內存查詢的結果輸出到該文件,WriteToLog 函數定義如下:

  

  int WriteToLog(char* str)

  {

   FILE* log;

   log = fopen(LOGFILE, "a+");

   if (log == NULL)

    return -1;

   f
   fclose(log);

   return 0;

  }

  聲明幾個全局變量,以便在程序的多個函數之間共享它們值。此外,做一個函數的前向定義:

  

  SERVICE_STATUS ServiceStatus;

  SERVICE_STATUS_HANDLE hStatus;

  

  void ServiceMain(int argc, char** argv);

  void ControlHandler(D
Word request);

  int InitService();

  現在,預備工作已經就緒,你可以開始編碼了。服務程序控制臺程序的一個子集。因此,開始你可以定義一個 main 函數,它是程序的入口點。對于服務程序來說,main 的代碼令人驚奇地簡短,因為它只創建分派表并啟動控制分派機。

  

  void main()

  {

   SERVICE_TABLE_ENTRY ServiceTable[2];

   ServiceTable[0].lpServiceName = "MemoryStatus";

   ServiceTable[0].lpServiceProc = (LPSERVICE_MAIN_FUNCTION)ServiceMain;

  

   ServiceTable[1].lpServiceName = NULL;

   ServiceTable[1].lpServiceProc = NULL;

  

   // 啟動服務的控制分派機線程

   StartServiceCtrlDispatcher(ServiceTable);

  }

  一個程序可能包含若干個服務。每一個服務都必須列于專門的分派表中(為此該程序定義了一個 ServiceTable 結構數組)。這個表中的每一項都要在 SERVICE_TABLE_ENTRY 結構之中。它有兩個域:

  lpServiceName: 指向表示服務名稱字符串的指針;當定義了多個服務時,那么這個域必須指定;

    lpServiceProc: 指向服務主函數的指針(服務入口點);

  分派表的最后一項必須是服務名和服務主函數域的 NULL 指針,文本例子程序中只宿主一個服務,所以服務名的定義是可選的。

  服務控制治理器(SCM:Services Control Manager)是一個治理系統所有服務的進程。當 SCM 啟動某個服務時,它等待某個進程的主線程來調用 StartServiceCtrlDispatcher 函數。將分派表傳遞給 StartServiceCtrlDispatcher。這將把調用進程的主線程轉換為控制分派器。該分派器啟動一個新線程,該線程運行分派表中每個服務的 ServiceMain 函數(本文例子中只有一個服務)分派器還監視程序中所有服務的執行情況。然后分派器將控制請求從 SCM 傳給服務。

  注重:假如 StartServiceCtrlDispatcher 函數30秒沒有被調用,便會報錯,為了避免這種情況,我們必須在 ServiceMain 函數中(參見本文例子)或在非主函數的單獨線程中初始化服務分派表。本文所描述的服務不需要防范這樣的情況。

  分派表中所有的服務執行完之后(例如,用戶通過“服務”控制面板程序停止它們),或者發生錯誤時。StartServiceCtrlDispatcher 調用返回。然后主進程終止。

  



推薦文章:搞笑之可愛水果表情  清新素潔水仙壁紙
更多文章 更多內容請看Windows操作系統安全集  Windows操作系統安裝  Windows頻道專題,或 第二步:ServiceMain 函數

  Listing 1 展示了 ServiceMain 的代碼。該函數是服務的入口點。它運行在一個單獨的線程當中,這個線程是由控制分派器創建的。ServiceMain 應該盡可能早早為服務注冊控制處理器。這要通過調用 RegisterServiceCtrlHadler 函數來實現。
你要將兩個參數傳遞給此函數:服務名和指向 ControlHandlerfunction 的指針。

  它指示控制分派器調用 ControlHandler 函數處理 SCM 控制請求。注冊完控制處理器之后,獲得狀態句柄(hStatus)。通過調用 SetServiceStatus 函數,用 hStatus 向 SCM 報告服務的狀態。

  Listing 1 展示了如何指定服務特征和其當前狀態來初始化 ServiceStatus 結構,ServiceStatus 結構的每個域都有其用途:

  dwServiceType:指示服務類型,創建 Win32 服務。賦值 SERVICE_WIN32;

  dwCurrentState:指定服務的當前狀態。因為服務的初始化在這里沒有完成,所以這里的狀態為 SERVICE_START_PENDING;

  dwControlsAccepted:這個域通知 SCM 服務接受哪個域。本文例子是答應 STOP 和 SHUTDOWN 請求。處理控制請求將在第三步討論;

  dwWin32ExitCode 和 dwServiceSpecificExitCode:這兩個域在你終止服務并報告退出細節時很有用。初始化服務時并不退出,因此,它們的值為 0;

  dwCheckPoint 和 dwWaitHint:這兩個域表示初始化某個服務進程時要30秒以上。本文例子服務的初始化過程很短,所以這兩個域的值都為 0。

  調用 SetServiceStatus 函數向 SCM 報告服務的狀態時。要提供 hStatus 句柄和 ServiceStatus 結構。注重 ServiceStatus 一個全局變量,所以你可以跨多個函數使用它。ServiceMain 函數中,你給結構的幾個域賦值,它們在服務運行的整個過程中都保持不變,比如:dwServiceType。

  在報告了服務狀態之后,你可以調用 InitService 函數來完成初始化。這個函數只是添加一個說明性字符串到日志文件。如下面代碼所示:

  

  // 服務初始化

  int InitService()

  {

   int result;

   result = WriteToLog("Monitoring started.");

   return(result);

  }

  在 ServiceMain 中,檢查 InitService 函數的返回值。假如初始化有錯(因為有可能寫日志文件失?。?,則將服務狀態置為終止并退出 ServiceMain:

  

  error = InitService();

  if (error)

  {

   // 初始化失敗,終止服務

   ServiceStatus.dwCurrentState = SERVICE_STOPPED;

   ServiceStatus.dwWin32ExitCode = -1;

   SetServiceStatus(hStatus, &ServiceStatus);

   // 退出 ServiceMain

   return;

  }

  假如初始化成功,則向 SCM 報告狀態:

  

  // 向 SCM 報告運行狀態

  ServiceStatus.dwCurrentState = SERVICE_RUNNING;

  SetServiceStatus (hStatus, &ServiceStatus);

  接著,啟動工作循環。每五秒鐘查詢一個可用物理內存并將結果寫入日志文件。

  如 Listing 1 所示,循環一直到服務的狀態為 SERVICE_RUNNING 或日志文件寫入出錯為止。狀態可能在 ControlHandler 函數響應 SCM 控制請求時修改。




推薦文章:搞笑之可愛水果表情  清新素潔水仙壁紙集
更多文章 更多內容請看Windows操作系統安全集  Windows操作系統安裝  Windows頻道專題,或 第三步:處理控制請求

  在第二步中,你用 ServiceMain 函數注冊了控制處理器函數??刂铺幚砥髋c處理各種 Windows 消息的窗口回調函數非常類似。它檢查 SCM 發送了什么請求并采取相應行動。


  每次你調用 SetServiceStatus 函數的時候,必須指定服務接收 STOP 和 SHUTDOWN 請求。Listing 2 示范了如何在 ControlHandler 函數中處理它們。

  STOP 請求是 SCM 終止服務的時候發送的。例如,假如用戶在“服務”控制面板中手動終止服務。SHUTDOWN 請求是關閉機器時,由 SCM 發送給所有運行中服務的請求。兩種情況的處理方式相同:

  寫日志文件,監視停止;

  向 SCM 報告 SERVICE_STOPPED 狀態;

  由于 ServiceStatus 結構對于整個程序而言為全局量,ServiceStatus 中的工作循環在當前狀態改變或服務終止后停止。其它的控制請求如:PAUSE 和 CONTINUE 在本文的例子沒有處理。

  控制處理器函數必須報告服務狀態,即便 SCM 每次發送控制請求的時候狀態保持相同。因此,不管響應什么請求,都要調用 SetServiceStatus。

  



  
C編寫Windows服務程序的五個步驟


  



  第四步:安裝和配置服務

  程序編好了,將之編譯成 exe 文件。本文例子創建的文件叫 MemoryStatus.exe,將它拷貝到 C:/MyServices 文件夾。為了在機器上安裝這個服務,需要用 SC.EXE 可執行文件,它是 Win32 Platform SDK 中附帶的一個工具。(譯者注:Visaul Studio .NET 2003 IDE 環境中也有這個工具,具體存放位置在:C:/Program Files/Microsoft Visual Studio .NET 2003/Common7/Tools/Bin/winnt)。使用這個實用工具可以安裝和移除服務。其它控制操作將通過服務控制面板來完成。以下是用命令行安裝 MemoryStatus 服務的方法:

  

  sc create MemoryStatus binpath= c:/MyServices/MemoryStatus.exe

  發出此創建命令。指定服務名和二進制文件的路徑(注重 binpath= 和路徑之間的那個空格)。安裝成功后,便可以用服務控制面板來控制這個服務(參見圖一)。用控制面板的工具欄啟動和終止這個服務。

  


推薦文章:搞笑之可愛水果表情  清新素潔水仙壁紙集
更多文章 更多內容請看Windows操作系統安全集  Windows操作系統安裝  Windows頻道專題,或
  
C編寫Windows服務程序的五個步驟

  



  MemoryStatus 的啟動類型是手動,也就是說根據需要來啟動這個服務。右鍵單擊該服務,然后選擇上下文菜單中的“屬性”菜單項,此時顯示該服務的屬性窗口。在這里可以修改啟動類型以及其它設置。你還可以從“常規”標簽中啟動/停止服務。以下是從系統中移除服務的方法:

  

  sc delete MemoryStatus

  指定 “delete” 選項和服務名。此服務將被標記為刪除,下次西通重啟后,該服務將被完全移除。

  第五步:測試服務

  從服務控制面板啟動 MemoryStatus 服務。假如初始化不出錯,表示啟動成功。過一會兒將服務停止。檢查一下 C:/MyServices 文件夾中 memstatus.txt 文件的服務輸出。在我的機器上輸出是這樣的:

  

  Monitoring started.

  273469440

  273379328

  273133568

  273084416

  Monitoring stopped.

  為了測試 MemoryStatus 服務在出錯情況下的行為,可以將 memstatus.txt 文件設置成只讀。
這樣一來,服務應該無法啟動。

  去掉只讀屬性,啟動服務,在將文件設成只讀。服務將停止執行,因為此時日志文件寫入失敗。假如你更新服務控制面板的內容,會發現服務狀態是已經停止。

  開發更大更好的服務程序

  理解 Win32 服務的基本概念,使你能更好地用 C++ 來設計包裝類。包裝類隱藏了對底層 Win32 函數的調用并提供了一種舒適的通用接口。修改 MemoryStatus 程序代碼,創建滿足自己需要的服務!為了實現比本文例子所示范的更復雜的任務,你可以創建多線程的服務,將作業劃分成幾個工作者線程并從 ServiceMain 函數中監視它們的執行。 更多文章 更多內容請看Windows操作系統安全集  Windows操作系統安裝  Windows頻道專題,或

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲色图美腿丝袜| 精品一区二区三区四区在线| 国产91成人在在线播放| 操91在线视频| 永久免费毛片在线播放不卡| 欧美第一淫aaasss性| 国产精品久久久久aaaa九色| 国产精品日本精品| 久久久久成人精品| 另类色图亚洲色图| 色偷偷91综合久久噜噜| 日韩成人激情在线| 日韩小视频在线观看| 色婷婷综合久久久久中文字幕1| 福利精品视频在线| 日韩精品在线观看网站| 91午夜理伦私人影院| 国产精品九九久久久久久久| 一个人看的www欧美| 欧美成人在线影院| 国产性色av一区二区| 青草青草久热精品视频在线观看| 日韩中文字幕视频在线| 在线a欧美视频| 美女久久久久久久久久久| 97超级碰碰人国产在线观看| 这里只有精品视频| 日本久久中文字幕| 日韩电影免费观看在线观看| 国产精品福利网站| 欧美午夜电影在线| 蜜臀久久99精品久久久久久宅男| 成人a免费视频| 欧美乱大交xxxxx| 伊人伊人伊人久久| 日本久久久久久久久久久| 欧美激情亚洲激情| 亚洲女人初尝黑人巨大| 日韩精品在线视频美女| 久久精品99久久久香蕉| 91免费人成网站在线观看18| 亚洲精品国产精品乱码不99按摩| 日韩精品在线观看一区二区| 国产精品一区二区三区毛片淫片| 亚洲一区二区三区四区在线播放| 亚洲男人天堂久| 亚洲国产婷婷香蕉久久久久久| 欧洲午夜精品久久久| 国产精品影院在线观看| 26uuu亚洲伊人春色| 亚洲三级免费看| 91免费电影网站| 日韩av免费观影| 日韩欧美国产激情| 美女久久久久久久| 亚洲人成电影网站色| www.日韩av.com| 久久综合久久88| 国产精品免费看久久久香蕉| 欧美成人午夜免费视在线看片| 黄色成人在线免费| 26uuu亚洲国产精品| 国产精品入口夜色视频大尺度| 78色国产精品| 黑人巨大精品欧美一区二区| 日韩av在线影视| 黑人欧美xxxx| 日韩小视频在线观看| 91禁国产网站| 国产精品黄色影片导航在线观看| 国产精品女主播视频| 欧美交受高潮1| 亚洲成人网在线| 亚洲精品国精品久久99热一| 97视频在线播放| 国产成人精品a视频一区www| 欧美激情videos| 亚洲欧美在线x视频| 欧美精品在线第一页| 国模精品视频一区二区三区| 日本一区二三区好的精华液| 欧美综合激情网| 美日韩精品免费视频| 亚洲偷熟乱区亚洲香蕉av| 亚洲精品乱码久久久久久按摩观| 国产自摸综合网| 国产精品久久久久久久久久久不卡| 亚州精品天堂中文字幕| 欧美美女15p| 国产精品久久久久久久久粉嫩av| 日韩高清欧美高清| 欧美又大粗又爽又黄大片视频| 国产精品黄视频| 亚洲一区二区免费在线| 精品女同一区二区三区在线播放| 国产成人一区二区三区| 992tv成人免费影院| 亚洲欧美在线第一页| 热久久免费视频精品| 97在线视频国产| 久久天天躁狠狠躁夜夜躁| 亚洲白拍色综合图区| 欧美在线视频导航| 欧美中文在线视频| 久久久999国产| 亚洲激情视频在线| 精品国产1区2区| 亚洲第一区中文99精品| 91tv亚洲精品香蕉国产一区7ujn| 91精品国产91久久久久久久久| 久青草国产97香蕉在线视频| 日韩av电影国产| 91美女片黄在线观看游戏| 色噜噜狠狠狠综合曰曰曰| 国产精品第一页在线| 国产成人在线一区| 国产日产欧美精品| 日本精品视频网站| 亚洲欧美一区二区三区在线| 欧美韩日一区二区| 久久久久国产一区二区三区| 色偷偷9999www| 青青久久aⅴ北条麻妃| 91在线观看免费| 成人黄色免费片| 国内精品久久影院| 中文字幕久热精品在线视频| 精品久久久一区| 亚洲国产精品女人久久久| 97在线视频免费看| 国产精品高潮粉嫩av| 精品国产成人av| 久久久久久久久国产精品| 日韩毛片中文字幕| 国产成人鲁鲁免费视频a| 欧美日韩激情视频| www亚洲欧美| 91九色视频导航| 欧美老女人性视频| 久久91亚洲精品中文字幕| 美女撒尿一区二区三区| 成人国产精品一区二区| 亚洲视频在线观看免费| 成人在线视频网站| 欧亚精品中文字幕| 欧美性高潮在线| 久久久精品久久久久| 亚洲在线视频观看| 久久久国产精品视频| 清纯唯美日韩制服另类| 亚洲欧美在线免费观看| 日韩精品免费在线| 日韩成人激情在线| 国产成人精品综合| 欧美高清不卡在线| 日韩电影第一页| 国产欧美va欧美va香蕉在| 欧美与欧洲交xxxx免费观看| 欧美丰满片xxx777| 国产亚洲精品成人av久久ww| 亚洲欧美日韩国产成人| 国产99视频在线观看| 午夜精品免费视频| 欧美高清性猛交|