首先,要控制windows services,是比較容易的事情,一堆現(xiàn)成的例子。
SYNtService 就是一個(gè)很好的例子。
要控制一個(gè)窗口退出是十分容易的事情,PostMessage就可以解決問(wèn)題。
要控制console退出,也有很多現(xiàn)成的方法。比較通用的方法:
1、TerminateProcess
最原始、最暴力的強(qiáng)制console退出的方法。console進(jìn)程毫無(wú)還手之力就over了。但我希望console在推出之前至少能處理一下“后事”。
2、signal / raise
原來(lái)windows也有signal,不過(guò)kill換成了raise。但是相對(duì)Unix系列的signal功能就差很遠(yuǎn)了。而且有一個(gè)麻煩的地方就是raise只能對(duì)本console生效,而不能對(duì)指定的process。
可以在signal里面指定一個(gè)call back函數(shù),在收到SIGINT/SIGTERM之類(lèi)的時(shí)候,處理一下事情,然后通知各個(gè)線程結(jié)束。
既然不能raise其他進(jìn)程,是否這個(gè)功能就不能用了呢?其實(shí)可以考慮一下CreateRemoteThread,然后在別人的進(jìn)程里面raise……
3、SetConsoleCtrlHandler / GenerateConsoleCtrlEvent
這個(gè)是console專(zhuān)門(mén)用來(lái)處理Ctrl-C/Ctrl-Break/以及windows關(guān)機(jī)事件等的處理方法。比signal更強(qiáng)大。而且說(shuō)明中寫(xiě)了,可以對(duì)其他的進(jìn)程進(jìn)行處理(還可以對(duì)進(jìn)程組處理)。用網(wǎng)上的話說(shuō)就是:很女子,很弓雖!
但是要注意,如果要對(duì)其創(chuàng)建的子進(jìn)程進(jìn)行處理的時(shí)候,創(chuàng)建子進(jìn)程必須要使用CREATE_NEW_PROCESS_GROUP標(biāo)志。另外一點(diǎn),文檔寫(xiě)的比較隱晦的就是,進(jìn)程必須要有console窗口。否則,調(diào)用GenerateConsoleCtrlEvent會(huì)返回6,說(shuō)ERROR_INVALID_HANDLE。
問(wèn)題來(lái)了,Service本身是沒(méi)有console窗口的,Service建立的子進(jìn)程就必須要自帶窗口了。但是一般為了美觀,Service啟動(dòng)的進(jìn)程都不想帶有窗口。那就變成了子進(jìn)程沒(méi)有console窗口,GenerateConsoleCtrlEvent失效了。
在網(wǎng)上查了很多資料(怎么沒(méi)看到很黃很暴力呢???),其中在
Louis K. Thomas
<louiNØSP@Msth@hotmÑOSP@Mail.coNÕSP@Mm> 的 SendSignal 提到一種做法,就是先獲得kernel32!CtrlRoutine的入口,然后通過(guò)CreateRemoteThread的方法,讓遠(yuǎn)程的console來(lái)執(zhí)行kernel32!CtrlRoutine。但這種方法有個(gè)問(wèn)題,在獲得kernel32!CtrlRoutine的時(shí)候,也是使用GenerateConsoleCtrlEvent來(lái)獲得。但是Service自己本身沒(méi)有console窗口,一調(diào)用GenerateConsoleCtrlEvent也是出錯(cuò)。
而另一篇
google討論組 文章,里面提到原來(lái)可以先AllocConsole、然后GenerateConsoleCtrlEvent、然后FreeConsole……于是解決方案就變成:
Service里面:
先AllocConsole
然后利用GenerateConsoleCtrlEvent獲得kernel32!CtrlRoutine
然后FreeConsole
當(dāng)需要結(jié)束進(jìn)程的時(shí)候,就調(diào)用CreateRemoteThread,把kernel32!CtrlRoutine的代碼注入到子process中執(zhí)行
這樣大家都看不到console窗口(service里面AllocConsole很快,看不到窗口出來(lái),甚至懷疑根本就有沒(méi)有窗口出來(lái)),同時(shí)又能通知子進(jìn)程優(yōu)雅地退出。
存在問(wèn)題:如果我的機(jī)器作為服務(wù)器啟動(dòng),即沒(méi)有進(jìn)入登錄狀態(tài),不知道這樣啟動(dòng)的Service會(huì)不會(huì)有問(wèn)題呢??暫時(shí)還沒(méi)有時(shí)間測(cè)試。