给Win32 GUI程序增加控制台窗口的方法

时间:2021-10-12 16:50:28

给Win32 GUI程序增加控制台窗口的方法

2008年10月11日 星期六 下午 04:43

在Win32的GUI程序中,没有控制台窗口,我们输出调试信息时有些不方便,以往我的做法是使用MessageBox弹出对话框、OutputDebugString向VS的调试窗口输出信息等,但总感觉不如控制台输出几句printf有效率。经过尝试发现GUI程序照样能够拥有(且仅能够拥有一个)控制台窗口,具体做法如下:

#include <locale.h>

在程序初始化时,加入以下代码:

FILE* fpDebugOut = NULL; 
FILE* fpDebugIn   = NULL; 
if( !AllocConsole() ) MessageBox(NULL, _T("控制台生成失败。"), NULL, 0); 
SetConsoleTitle(_T("Debug Window")); 
_tfreopen_s(&fpDebugOut, _T("CONOUT$"),_T("w"), stdout); 
_tfreopen_s(&fpDebugIn, _T("CONIN$"), _T("r"), stdin); 
_tsetlocale(LC_ALL, _T("chs"));     //这是必要的,否则unicode模式下使用C库函数控制台输出不了中文 WriteConsole函数不受影响

注意AllocConsole可能失败,原因可能是程序已经于一个控制台窗口关联过了,等等。

程序结束时,加入以下代码:

fclose(fpDebugOut); 
fclose(fpDebugIn); 
FreeConsole();

好,现在已经加入了控制台,可以输出字符串了,使用标准的C库函数或者WriteConsole都可以:

使用WriteConsole的话前面其实不用调用freopen重定向输出流了,直接WriteConsole就行:

TCHAR str[256]; 
    DWORD numOfCharWritten = 0; 
    _stprintf_s(str, 256, _T("今年是 %d 年\n"), 2008); 
    WriteConsole( GetStdHandle(STD_OUTPUT_HANDLE), str, _tcslen(str), &numOfCharWritten, NULL);

使用C库函数则更简单,但上面一定要调用freopen才有效:

int tmp = 0; 
_tscanf_s(_T("%d\n"), &tmp); 
_tprintf_s(_T("刚才输入的是 %d\n"), tmp);

比如在响应WM_PAINT消息时,加入_tprintf_s(_T("重绘 %d 次\n"), ++count),则每次窗口绘制的时候都将输出总重绘次数。

最后,定义两个宏图省事:

#define CREATE_DEBUG_CONSOLE \ 
FILE* fpDebugOut = NULL; \ 
FILE* fpDebugIn = NULL; \ 
FILE* fpDebugErr = NULL; \ 
if( !AllocConsole() ) \ 
   MessageBox(NULL, _T("控制台生成失败。"), NULL, 0); \ 
SetConsoleTitle(_T("Debug Window")); \ 
_tfreopen_s(&fpDebugOut, _T("CONOUT$"),_T("w"), stdout); \ 
_tfreopen_s(&fpDebugIn, _T("CONIN$"), _T("r"), stdin); \ 
_tfreopen_s(&fpDebugErr, _T("CONOUT$"),_T("w"), stderr); \ 
_tsetlocale(LC_ALL, _T("chs"));

#define RELEASE_DEBUG_CONSOLE \ 
fclose(fpDebugOut); \ 
fclose(fpDebugIn); \ 
fclose(fpDebugErr); \ 
FreeConsole();