青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

TGeek

Technophile
posts - 1, comments - 0, trackbacks - 0, articles - 1
  C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理
第一步:主函數和全局定義

首先,包含所需的頭文件。例子要調用 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;
????fprintf(log,?
"%s\n",?str);
????fclose(log);
????
return?0;
}


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

SERVICE_STATUS ServiceStatus;
SERVICE_STATUS_HANDLE hStatus;

void ServiceMain(int argc, char** argv);
void ControlHandler(DWORD 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 調用返回。然后主進程終止。


第二步: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;?
}
?

?


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            国产免费成人在线视频| 韩国视频理论视频久久| 亚洲视频综合| 在线亚洲+欧美+日本专区| 欧美日韩1区2区3区| 亚洲深夜激情| 亚洲一区二区视频在线| 国产欧美一区二区三区在线看蜜臀| 亚洲一区免费看| 午夜精品一区二区三区在线播放| 国产色综合网| 免费不卡在线观看av| 欧美福利视频网站| 亚洲一区二区三区在线观看视频| 亚洲午夜激情网页| 狠狠操狠狠色综合网| 亚洲福利视频三区| 欧美午夜精品| 久久人人97超碰人人澡爱香蕉| 久久久噜噜噜久噜久久| 一区二区三区偷拍| 性欧美精品高清| 亚洲精品欧美极品| 亚洲字幕一区二区| 在线观看中文字幕亚洲| 亚洲激情自拍| 国产欧美一区视频| 亚洲国内自拍| 国产亚洲欧美色| 亚洲精品国产精品久久清纯直播| 国产精品三级久久久久久电影| 老司机成人网| 国产精品国产三级国产aⅴ无密码 国产精品国产三级国产aⅴ入口 | 久久综合色影院| 欧美伦理在线观看| 久久久五月天| 国产精品成人免费精品自在线观看| 欧美一级视频精品观看| 欧美华人在线视频| 欧美三级电影精品| 久久亚洲图片| 国产精品久久一区主播| 亚洲电影免费观看高清完整版| 国产伦理一区| 一本一本a久久| 亚洲精品五月天| 久久精品亚洲精品国产欧美kt∨| 亚洲婷婷综合久久一本伊一区| 久久婷婷久久| 欧美影院一区| 国产精品av免费在线观看| 美女脱光内衣内裤视频久久影院| 国产精品国产三级国产专播品爱网| 欧美大香线蕉线伊人久久国产精品| 国产精品日本| 一本色道久久综合狠狠躁篇的优点 | 免费不卡视频| 久久人人爽人人| 国产日韩亚洲欧美| 亚洲尤物视频网| 亚洲欧美大片| 欧美亚洲不卡| 亚洲小视频在线观看| 亚洲午夜精品福利| 欧美亚洲第一页| 一本色道久久加勒比精品| 中文精品视频| 国产精品hd| 日韩一级在线观看| 亚洲视频第一页| 欧美私人网站| 亚洲一品av免费观看| 亚洲欧美国产77777| 国产精品乱码一区二三区小蝌蚪| 亚洲手机成人高清视频| 午夜精品免费视频| 国产欧美欧美| 久久久国产成人精品| 久久一区激情| 亚洲国产一区二区三区在线播 | 香蕉久久夜色| 久久久久久久久久久久久9999| 国产一区二区精品久久99| 久久久久国内| 亚洲欧洲日产国产网站| 亚洲欧美大片| 国产在线拍揄自揄视频不卡99| 久久精品视频在线观看| 欧美成人嫩草网站| av不卡在线| 国产精品久久999| 欧美一区二区在线免费观看| 猫咪成人在线观看| 日韩午夜免费| 国产麻豆视频精品| 蜜桃av综合| 中国成人黄色视屏| 久久综合九色综合欧美就去吻| 亚洲午夜电影| 欧美插天视频在线播放| 一本一道久久综合狠狠老精东影业 | 免费人成网站在线观看欧美高清 | 亚洲性视频网址| 国产在线播精品第三| 欧美成人国产一区二区| 一区二区av在线| 蜜臀va亚洲va欧美va天堂| 在线视频亚洲一区| 黄色免费成人| 国产精品qvod| 媚黑女一区二区| 亚洲综合二区| 亚洲精品一区二区三区福利| 久久精品国产69国产精品亚洲| 亚洲欧洲精品一区二区三区不卡| 国产精品毛片a∨一区二区三区| 久久久久久久精| 亚洲私人影吧| 91久久久久久久久| 久久免费精品视频| 亚洲曰本av电影| 亚洲人成亚洲人成在线观看| 国产一区二区三区四区三区四| 欧美精品久久久久久久免费观看| 欧美一区二区视频在线观看| 日韩一级精品| 亚洲高清久久网| 久久伊伊香蕉| 久久久久久久久久久久久9999| 亚洲天堂网在线观看| 亚洲精品一区二区三区四区高清| 精久久久久久久久久久| 国产伦精品一区二区三区在线观看| 欧美日韩国产色视频| 欧美成人免费全部| 久久综合999| 久久免费观看视频| 欧美亚洲视频在线观看| 亚洲在线视频网站| 一区二区日韩| 9人人澡人人爽人人精品| 亚洲日本va在线观看| 欧美激情无毛| 亚洲电影在线观看| 亚洲国产一区二区三区青草影视| 久热爱精品视频线路一| 久久久久久久999精品视频| 亚洲欧美日韩国产一区二区| 亚洲一区欧美一区| 亚洲在线成人| 欧美一区二区三区四区在线| 欧美一二三区精品| 久久精品视频导航| 美女视频一区免费观看| 免费看精品久久片| 亚洲国产一区二区三区在线播| 亚洲成色www久久网站| 亚洲国产精品一区二区第一页 | 免费观看成人www动漫视频| 久久婷婷色综合| 欧美 亚欧 日韩视频在线| 国产精品久久久久久户外露出| 欧美另类高清视频在线| 欧美日韩一区免费| 国产精品夫妻自拍| 国产日韩欧美在线| 亚洲黄一区二区三区| 亚洲精品在线一区二区| 亚洲视频专区在线| 欧美专区福利在线| 欧美粗暴jizz性欧美20| 日韩午夜免费视频| 午夜一区二区三区不卡视频| 久久久久久久网| 嫩草国产精品入口| 国产精品v日韩精品| 国产日韩精品视频一区二区三区| 国产自产女人91一区在线观看| 亚洲第一成人在线| 亚洲永久精品国产| 久久嫩草精品久久久精品一| 亚洲国产精品女人久久久| 亚洲午夜视频在线观看| 久久亚洲国产精品一区二区| 欧美日韩在线播放一区二区| 国产日韩一级二级三级| 亚洲免费观看高清在线观看| 亚洲综合视频一区| 欧美成人日本| 午夜欧美精品久久久久久久| 蜜桃av久久久亚洲精品| 国产精品推荐精品| 亚洲三级色网| 久久久久久网址| 一本色道久久综合亚洲精品不 | 免费成人网www| 国产精品一级二级三级| 亚洲毛片在线看| 久久久久久久激情视频| 亚洲午夜精品视频|