我不知道各位在编程过程中是用什么方法调试程序的,不过象我这种不会用VC调试器的人只好用“MessageBox”什么的来进行跟踪了,要是碰到麻烦一点的,只好用ollydbg来调试了。与其花时间去搞懂VC下的那些“ASSERT”、“VERIFY”还不如自己写个调试器来得实惠。
VC调试器我没用过,所以也不能评论他的优劣,但是象跟踪变量这样的功能用“MessageBox”真的就可以搞定了(要是碰到个循环就麻烦了点:D),复杂的问题还是用ollydbg来得直观(我不是说VC调试器不直观,主要是我用不来)。在有些情况下,debug版没有问题并不代表release版也没有问题,所以我向来是在release版下进行调试的。这样一来VC调试器无法进行源代码级的调试,那就不占任何优势了。
后来,我终于决定摆脱使用“MessageBox”这种丑陋方法,决定想一个方便、简洁、易用的方法。这个方法必须:1、能方便地输出调试信息;2、而且要可以在执行过程中进行干预;3、最重要的一点是“调试版”要能很方便地修改成“release版”,最好是无需修改源代码。想来想去还是用命令行窗口实现最简单,现成的CRT函数——printf和scanf就能实现格式化的输出和输入,第一条和第二条就可以实现了,而且不用写一行代码。至于第三条,我考虑用宏来实现。
定义一个“DEBUG_FLAG”标志来控制编译目标:
#if defined(DEBUG_FLAG) #pragma comment(linker,"/subsystem:console") #else #pragma comment(linker,"/subsystem:windows") #endif
上面的代码意思是:如果定义了“DEBUG_FALG”标志,就将目标编译为控制台程序。本来是想使用“AllocConsole”函数来创建一个命令行窗口的,但是“printf”函数无法在这个新创建的命令行窗口上输出任何信息(用“WriteConsole”函数却可以,奇怪啊~)。
下面是一个Sample:
#define DEBUG_FLAG #include #pragma comment(linker,"/entry:WinEntry") #if defined(DEBUG_FLAG) #pragma comment(linker,"/subsystem:console") #else #pragma comment(linker,"/subsystem:windows") #endif
#if defined(DEBUG_FLAG) _CRTIMP int (__cdecl *printf)(const char *, ...); _CRTIMP int (__cdecl *scanf)(const char *, ...); #endif
void WinEntry(void) { DWORD time; #if defined(DEBUG_FLAG) char key; HMODULE hMsvcrt=LoadLibrary("msvcrt.dll"); printf=(void *)GetProcAddress(hMsvcrt,"printf"); scanf=(void *)GetProcAddress(hMsvcrt,"scanf"); SetConsoleTitle("调试窗口"); printf("程序成功加载...\n是否要继续执行?(Y/N)"); scanf("%1c",&key); if(!(key=='Y'||key=='y')) ExitProcess(0); #endif time=GetTickCount(); #if defined(DEBUG_FLAG) printf("已经开机:%d",time); #endif //some code here ExitProcess(0); } 
|