vc控制臺程序關(guān)閉事件時的處理方式及注意點詳解
百度可以找到很多關(guān)于這個問題解決的方法
關(guān)鍵控制臺API函數(shù):SetConsoleCtrlHandler
在支持C++ 11以上的編譯器中,你可以這么做。
SetConsoleCtrlHandler([](DWORD fdwctrltype)->BOOL { if (fdwctrltype == CTRL_CLOSE_EVENT) { // 你的善后代碼... return TRUE; } return FALSE; }, TRUE);
最初這么做是很舒服的,但之后發(fā)現(xiàn)了問題:
Windows控制臺在標記狀態(tài)下,printf之類的輸出函數(shù),會阻塞在標記選擇時(點控制臺左上角-編輯-標記)。
這就導(dǎo)致了,我們的善后代碼中,可能會死鎖,例如你要優(yōu)雅的結(jié)束一個線程,這個線程在最后的時候printf了。
線程里printf等待標記狀態(tài),SetConsoleCtrlHandler回調(diào)函數(shù)里等待線程結(jié)束,總之就是死鎖。
我本來想著,去找到能夠獲取這種標記狀態(tài)的控制臺API,但找了很久都沒有結(jié)果。
最后,我就考慮,有沒有方法讓printf不與標記狀態(tài)發(fā)生死鎖,答案是: 輸出流重定向。
所以,代碼變成這樣:
SetConsoleCtrlHandler([](DWORD fdwctrltype)->BOOL { if (fdwctrltype == CTRL_CLOSE_EVENT) { char szbuf[0x1000]; setvbuf(stdout, szbuf, _IOFBF, 0x1000); // 你的善后代碼... return TRUE; } return FALSE; }, TRUE);
這么做之后,世界果然更美好了,如果最后這些日志信息對你來說是重要的,那么你可能需要寫更多代碼去實現(xiàn)。
知識點擴展:
實例:
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控制臺程序關(guān)閉事件時的處理方式及注意點詳解的文章就介紹到這了,更多相關(guān)vc控制臺程序關(guān)閉事件時的正確處理方式內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言開發(fā)實現(xiàn)井字棋及電腦落子優(yōu)化示例詳解
以前上課經(jīng)常和同桌玩起井字棋,那么我們就當(dāng)我們回憶童年,現(xiàn)在也用C語言來實現(xiàn)井字棋,本次代碼相對于初階的井字棋,在電腦下棋代碼部分做了優(yōu)化,使得電腦更加具有威脅2021-11-11大家注意vector, list, set, map成員函數(shù)erase
set和map是由紅黑樹來實現(xiàn)的,當(dāng)erase的時候迭代器就失效了,也就是說我們要在迭代器失效之前保留一個副本,根據(jù)這個副本我們才能繼續(xù)遍歷下一個元素2013-09-09C++ Qt開發(fā)之PushButton按鈕組件的使用詳解
Qt 是一個跨平臺C++圖形界面開發(fā)庫,利用Qt可以快速開發(fā)跨平臺窗體應(yīng)用程序,本文將重點介紹QPushButton按鈕組件的常用方法及靈活運用,感興趣的小伙伴可以學(xué)習(xí)一下2023-12-12