• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            TGeek

            Technophile
            posts - 1, comments - 0, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理
            一個程序可能包含若干個服務。每一個服務都必須列于專門的分派表中(為此該程序定義了一個 ServiceTable 結構數組)。這個表中的每一項都要在 SERVICE_TABLE_ENTRY 結構之中。它有兩個域:

            lpServiceName: 指向表示服務名稱字符串的指針;當定義了多個服務時,那么這個域必須指定;
            lpServiceProc: 指向服務主函數的指針(服務入口點);
              分派表的最后一項必須是服務名和服務主函數域的 NULL 指針,文本例子程序中只宿主一個服務,所以服務名的定義是可選的。
              服務控制管理器(SCM:Services Control Manager)是一個管理系統所有服務的進程。當 SCM 啟動某個服務時,它等待某個進程的主線程來調用 StartServiceCtrlDispatcher 函數。將分派表傳遞給 StartServiceCtrlDispatcher。這將把調用進程的主線程轉換為控制分派器。該分派器啟動一個新線程,該線程運行分派表中每個服務的 ServiceMain 函數(本文例子中只有一個服務)分派器還監視程序中所有服務的執行情況。然后分派器將控制請求從 SCM 傳給服務。

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

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


            第二步: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 控制請求時修改。


            第三步:處理控制請求

              在第二步中,你用 ServiceMain 函數注冊了控制處理器函數。控制處理器與處理各種 Windows 消息的窗口回調函數非常類似。它檢查 SCM 發送了什么請求并采取相應行動。
              每次你調用 SetServiceStatus 函數的時候,必須指定服務接收 STOP 和 SHUTDOWN 請求。Listing 2 示范了如何在 ControlHandler 函數中處理它們。
              STOP 請求是 SCM 終止服務的時候發送的。例如,如果用戶在“服務”控制面板中手動終止服務。SHUTDOWN 請求是關閉機器時,由 SCM 發送給所有運行中服務的請求。兩種情況的處理方式相同:

            寫日志文件,監視停止;
            向 SCM 報告 SERVICE_STOPPED 狀態;
              由于 ServiceStatus 結構對于整個程序而言為全局量,ServiceStatus 中的工作循環在當前狀態改變或服務終止后停止。其它的控制請求如:PAUSE 和 CONTINUE 在本文的例子沒有處理。
              控制處理器函數必須報告服務狀態,即便 SCM 每次發送控制請求的時候狀態保持相同。因此,不管響應什么請求,都要調用 SetServiceStatus。

            第四步:安裝和配置服務

              程序編好了,將之編譯成 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= 和路徑之間的那個空格)。安裝成功后,便可以用服務控制面板來控制這個服務。用控制面板的工具欄啟動和終止這個服務。

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

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

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

            Monitoring started.
            273469440
            273379328
            273133568
            273084416
            Monitoring stopped.
              為了測試 MemoryStatus 服務在出錯情況下的行為,可以將 memstatus.txt 文件設置成只讀。這樣一來,服務應該無法啟動。
              去掉只讀屬性,啟動服務,在將文件設成只讀。服務將停止執行,因為此時日志文件寫入失敗。如果你更新服務控制面板的內容,會發現服務狀態是已經停止。

            #include?<windows.h>
            #include?
            <stdio.h>

            #define?SLEEP_TIME?
            5000
            #define?LOGFILE?
            "C:\\MyServices\\memstatus.txt"

            SERVICE_STATUS?ServiceStatus;?
            SERVICE_STATUS_HANDLE?hStatus;?
            ?
            void??ServiceMain(int?argc,?char**?argv);?
            void??ControlHandler(DWORD?request);?
            int?InitService();

            int?WriteToLog(char*?str)
            {
            ????FILE
            *?log;
            ????log?
            =?fopen(LOGFILE,?"a+");
            ????
            if?(log?==?NULL)
            ????????
            return?-1;
            ????fprintf(log,?
            "%s\n",?str);
            ????fclose(log);
            ????
            return?0;
            }


            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;
            ????
            //?Start?the?control?dispatcher?thread?for?our?service
            ????StartServiceCtrlDispatcher(ServiceTable);??
            }



            void?ServiceMain(int?argc,?char**?argv)?
            {?
            ????
            int?error;?
            ?
            ????ServiceStatus.dwServiceType????????
            =?SERVICE_WIN32;?
            ????ServiceStatus.dwCurrentState???????
            =?SERVICE_START_PENDING;?
            ????ServiceStatus.dwControlsAccepted???
            =?SERVICE_ACCEPT_STOP?|?SERVICE_ACCEPT_SHUTDOWN;
            ????ServiceStatus.dwWin32ExitCode??????
            =?0;?
            ????ServiceStatus.dwServiceSpecificExitCode?
            =?0;?
            ????ServiceStatus.dwCheckPoint?????????
            =?0;?
            ????ServiceStatus.dwWaitHint???????????
            =?0;?
            ?
            ????hStatus?
            =?RegisterServiceCtrlHandler(
            ????????
            "MemoryStatus",?
            ????????(LPHANDLER_FUNCTION)ControlHandler);?
            ????
            if?(hStatus?==?(SERVICE_STATUS_HANDLE)0)?
            ????
            {?
            ????????
            //?Registering?Control?Handler?failed
            ????????return;?
            ????}
            ??
            ????
            //?Initialize?Service?
            ????error?=?InitService();?
            ????
            if?(error)?
            ????
            {
            ????????
            //?Initialization?failed
            ????????ServiceStatus.dwCurrentState???????=?SERVICE_STOPPED;?
            ????????ServiceStatus.dwWin32ExitCode??????
            =?-1;?
            ????????SetServiceStatus(hStatus,?
            &ServiceStatus);?
            ????????
            return;?
            ????}
            ?
            ????
            //?We?report?the?running?status?to?SCM.?
            ????ServiceStatus.dwCurrentState?=?SERVICE_RUNNING;?
            ????SetServiceStatus?(hStatus,?
            &ServiceStatus);
            ?
            ????MEMORYSTATUS?memory;
            ????
            //?The?worker?loop?of?a?service
            ????while?(ServiceStatus.dwCurrentState?==?SERVICE_RUNNING)
            ????
            {
            ????????
            char?buffer[16];
            ????????GlobalMemoryStatus(
            &memory);
            ????????sprintf(buffer,?
            "%d",?memory.dwAvailPhys);
            ????????
            int?result?=?WriteToLog(buffer);
            ????????
            if?(result)
            ????????
            {
            ????????????ServiceStatus.dwCurrentState???????
            =?SERVICE_STOPPED;?
            ????????????ServiceStatus.dwWin32ExitCode??????
            =?-1;?
            ????????????SetServiceStatus(hStatus,?
            &ServiceStatus);
            ????????????
            return;
            ????????}


            ????????Sleep(SLEEP_TIME);
            ????}

            ????
            return;?
            }

            ?
            //?Service?initialization
            int?InitService()?
            {?
            ????
            int?result;
            ????result?
            =?WriteToLog("Monitoring?started.");
            ????
            return(result);?
            }
            ?

            //?Control?handler?function
            void?ControlHandler(DWORD?request)?
            {?
            ????
            switch(request)?
            ????
            {?
            ????????
            case?SERVICE_CONTROL_STOP:?
            ?????????????WriteToLog(
            "Monitoring?stopped.");

            ????????????ServiceStatus.dwWin32ExitCode?
            =?0;?
            ????????????ServiceStatus.dwCurrentState??
            =?SERVICE_STOPPED;?
            ????????????SetServiceStatus?(hStatus,?
            &ServiceStatus);
            ????????????
            return;?
            ?
            ????????
            case?SERVICE_CONTROL_SHUTDOWN:?
            ????????????WriteToLog(
            "Monitoring?stopped.");

            ????????????ServiceStatus.dwWin32ExitCode?
            =?0;?
            ????????????ServiceStatus.dwCurrentState??
            =?SERVICE_STOPPED;?
            ????????????SetServiceStatus?(hStatus,?
            &ServiceStatus);
            ????????????
            return;?
            ????????
            ????????
            default:
            ????????????
            break;
            ????}
            ?
            ?
            ????
            //?Report?current?status
            ????SetServiceStatus?(hStatus,??&ServiceStatus);
            ?
            ????
            return;?
            }
            ?

            久久成人精品| 91精品国产高清久久久久久国产嫩草 | 无码国内精品久久人妻蜜桃| 一本色综合久久| 久久免费看黄a级毛片| 久久se精品一区精品二区| 色综合久久88色综合天天 | 国内精品人妻无码久久久影院导航| 国产V综合V亚洲欧美久久| 精品乱码久久久久久夜夜嗨| 久久久久久综合网天天| 久久精品无码一区二区三区| 久久婷婷五月综合色99啪ak| 久久A级毛片免费观看| 久久天天躁狠狠躁夜夜不卡| 久久96国产精品久久久| 亚洲午夜无码久久久久| 久久亚洲国产成人精品无码区| 乱亲女H秽乱长久久久| 一本久久a久久精品综合香蕉| 91久久国产视频| 久久精品人人做人人爽97 | 亚洲欧美一区二区三区久久| 久久精品无码一区二区无码| 精品久久久久成人码免费动漫 | 亚洲а∨天堂久久精品9966| 91久久精一区二区三区大全| 无码任你躁久久久久久老妇App| 亚洲国产精品久久久久| 久久精品国产亚洲av水果派| 国产成人精品综合久久久久 | 久久无码人妻一区二区三区午夜 | 久久国产免费观看精品3| 久久狠狠爱亚洲综合影院| 亚洲欧美国产精品专区久久 | 亚洲国产精品久久电影欧美| 亚洲日本久久久午夜精品| 怡红院日本一道日本久久 | 色婷婷噜噜久久国产精品12p| 国产精品欧美久久久久无广告| 99久久久国产精品免费无卡顿|