百度可以找到很多關(guān)于這個(gè)問(wèn)題解決的方法
關(guān)鍵控制臺(tái)API函數(shù):SetConsoleCtrlHandler
在支持C++ 11以上的編譯器中,你可以這么做。
1
2
3
4
5
6
7
|
SetConsoleCtrlHandler([]( DWORD fdwctrltype)-> BOOL { if (fdwctrltype == CTRL_CLOSE_EVENT) { // 你的善后代碼... return TRUE; } return FALSE; }, TRUE); |
最初這么做是很舒服的,但之后發(fā)現(xiàn)了問(wèn)題:
Windows控制臺(tái)在標(biāo)記狀態(tài)下,printf之類(lèi)的輸出函數(shù),會(huì)阻塞在標(biāo)記選擇時(shí)(點(diǎn)控制臺(tái)左上角-編輯-標(biāo)記)。
這就導(dǎo)致了,我們的善后代碼中,可能會(huì)死鎖,例如你要優(yōu)雅的結(jié)束一個(gè)線(xiàn)程,這個(gè)線(xiàn)程在最后的時(shí)候printf了。
線(xiàn)程里printf等待標(biāo)記狀態(tài),SetConsoleCtrlHandler回調(diào)函數(shù)里等待線(xiàn)程結(jié)束,總之就是死鎖。
我本來(lái)想著,去找到能夠獲取這種標(biāo)記狀態(tài)的控制臺(tái)API,但找了很久都沒(méi)有結(jié)果。
最后,我就考慮,有沒(méi)有方法讓printf不與標(biāo)記狀態(tài)發(fā)生死鎖,答案是: 輸出流重定向。
所以,代碼變成這樣:
1
2
3
4
5
6
7
8
9
|
SetConsoleCtrlHandler([]( DWORD fdwctrltype)-> BOOL { if (fdwctrltype == CTRL_CLOSE_EVENT) { char szbuf[0x1000]; setvbuf (stdout, szbuf, _IOFBF, 0x1000); // 你的善后代碼... return TRUE; } return FALSE; }, TRUE); |
這么做之后,世界果然更美好了,如果最后這些日志信息對(duì)你來(lái)說(shuō)是重要的,那么你可能需要寫(xiě)更多代碼去實(shí)現(xiàn)。
知識(shí)點(diǎn)擴(kuò)展:
實(shí)例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
BOOL WINAPI ConsoleHandler( DWORD CEvent) { DWORD e = 0; switch (CEvent) { case CTRL_C_EVENT: e = CTRL_C_EVENT; break ; case CTRL_BREAK_EVENT: e = CTRL_BREAK_EVENT; break ; case CTRL_CLOSE_EVENT: e = CTRL_CLOSE_EVENT; break ; case CTRL_LOGOFF_EVENT: break ; case CTRL_SHUTDOWN_EVENT: break ; } return true ; } int main( int argc, char * argv[]) { if (SetConsoleCtrlHandler((PHANDLER_ROUTINE)ConsoleHandler, TRUE) == FALSE) { //安裝失敗 return -1; } GenerateConsoleCtrlEvent(CTRL_C_EVENT, 0); //手工產(chǎn)生一事件 } |
到此這篇關(guān)于vc控制臺(tái)程序關(guān)閉事件時(shí)的處理方式及注意點(diǎn)詳解的文章就介紹到這了,更多相關(guān)vc控制臺(tái)程序關(guān)閉事件時(shí)的正確處理方式內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://www.cnblogs.com/babypapa/p/13032603.html