上一节,我们讲解了 assert 断言函数的使用,本节我们来学习在调试器的调试窗口上输出我们自己的调试信息,在这里,我们将用到一个 Windows 操作系统提供的函数 —— OutputDebugString,这个函数非常常用,他可以向调试输出窗口输出信息(无需设置断点,执行就会输出调试信息),并且一般只在绑定了调试器的情况下才会生效,否则会被 Windows 直接忽略。接下来我们了解一下这个函数的使用方法。
首先,这个函数在 windows.h 中被定义,所以我们需要包含 windows.h 这个头文件才能使用 OutputDebugString 函数。这个函数的使用方式非常的简单,它只有简单的一个参数——我们要输出的调试信息。但是有一点值得注意:准确来说 OutputDebugString 并不是一个函数,他是一个宏。在高版本的 Visual Studio 中,因为编译的时候 Visual Studio 默认定义了 UNICODE 宏,所以我们查找 OutputDebugString 的定义会看到如下代码:
#ifdef UNICODE
#define OutputDebugString OutputDebugStringW
#else
#define OutputDebugString OutputDebugStringA
#endif // !UNICODE
我们可以从代码高亮上看到,OutputDebugString 实际上等价于 OutputDebugStringW,这就意味着我们必须传入宽字符串(事实上只要定义了 UNICODE ,调用所有 Windows 提供的函数都需要使用宽字符),或者使用 TEXT 或 _T 宏,并且这是最好的方法,这个宏会自动识别编译器是否处于默认宽字符的状态并对传入字符串做些处理,使用这个宏可以加强代码对不同编译器不同编译参数的兼容性。下面我们就来看一段示例代码:
#include <windows.h> //使用 OutputDebugString 包含此文件
#include <tchar.h> //使用 TEXT 宏需要包含此文件
int main(){
OutputDebugString(TEXT("你好,城东书院。"));
OutputDebugString(_T("大家好才是真的好。"));
//也可以:OutputDebugStringA("大家好才是真的好。");
//也可以:OutputDebugStringW(L"大家好才是真的好。");
//使用自动字符宏 TEXT 或者 _T 可以自动判断是否使用宽字符
system("pause"); //使程序暂停一下
return 0;
}
在程序执行 system("pause"); 暂停的时候我们来观察一下我们的,调试输出窗口:
怎么样,是不是输出了“你好,城东书院。大家好才是真的好。”呢?这个函数与 printf 等函数一样,需要我们自己插入换行符,但在 Windows 下,我们一般使用 \r\n 作为完整的换行符。直接使用这个调试信息输出函数有个弊端,那就是它不能直接输出含参数的字符串。但是我们可以通过 sprintf / wsprintf 等缓冲区写入函数来间接实现输出含参数字符串的功能。下面是一段示例代码:
#include <stdio.h>
#include <windows.h>
int main(){
//注意!这段代码我们指定使用ANSI字符!
char szBuffer[200];
int number = 100;
sprintf_s(szBuffer, "变量 number 的值是 %d \r\n", number); //写入缓冲区,注意不要溢出
OutputDebugStringA(szBuffer);
sprintf_s(szBuffer, "变量 number 的地址是 %x \r\n", &number);
OutputDebugStringA(szBuffer);
//我门指定使用 ANSI 版本的 OutputDebugString 函数
return 0;
}
我们按 F5 开始调试:
我们看到了输出的结果,怎么样,大家是不是觉得这样调用这个函数很麻烦?为解决此问题,这里为大家提供了一个更好的解决方案,我们可以自己写一个前端函数,然后保存到头文件中(编译生成 dll 也可以,有兴趣的同学可以试试)。为了方便,我们已经编写好了这么一套函数。代码如下:
#include <stdio.h>
#include <windows.h>
#ifndef _DEBUG_INFO_HEADER_
//防止头文件被重复载入出错
#define _DEBUG_INFO_HEADER_
#if (defined UNICODE)||(defined _UNICODE)
#define DebugInfo DebugInfoW
#else
#define DebugInfo DebugInfoA
#endif
// 函数: DebugInfoA(char*, int, ...)
//
// 目的: 以窄字符的形式输出调试信息
//
// char* str - 格式化 ANSI 字符串
// ... - 任意不定长参数
//
void DebugInfoA(char* str, ...){
char szBuffer[500]; //注意不要让缓冲区溢出!
va_list Argv;
va_start(Argv, str);
_vsnprintf_s(szBuffer, 500, str, Argv);
va_end(Argv);
OutputDebugStringA(szBuffer);
}
// 函数: DebugInfoW(char*, int, ...)
//
// 目的: 以宽字符的形式输出调试信息
//
// char* str - 格式化 UNICODE 字符串
// ... - 任意不定长参数
//
void DebugInfoW(wchar_t* str, ...){
wchar_t szBuffer[1000];
va_list Argv;
va_start(Argv, str);
_vsnwprintf_s(szBuffer, 500, str, Argv);
va_end(Argv);
OutputDebugStringW(szBuffer);
}
#endif
上面的这段代码会自动识别编译器是否默认使用了宽字符并且使用对应版本的输出函数,其中注释为 Visual Studio 的智能提示信息,我们把上面的代码保存到 debuginfo.h 并添加到当前工程中,就可以直接通过如下代码调用:
#include <stdio.h>
#include <tchar.h>
#include <windows.h>
#include "debuginfo.h"
int main(){
int num;
//这里我们使用微软提供的 xxxx_s 安全函数
printf_s("请输入数值:\n");
scanf_s("%d", &num);
DebugInfo(TEXT("用户输入的数是 %d\n"), num);
return 0;
}
我们按 F5 开始调试,我们输入666,观察 IDE 的输出窗口:
我们可以看到,它正确地输出了 “用户输入的数是 666” ,这就说明我们的代码执行成功了,上面的代码大家可以随意修改我们上面提供的代码以符合自己的实际需要。但是请注意不要让缓冲区溢出,否则会造成不可估计的后果。
我们除了在调试器中可以看到调试字符串的输出,我们还可以借助 Sysinternals 软件公司研发的一个相当高级的工具 —— DebugView 调试信息捕捉工具,这个工具可以在随时随地捕捉 OutputDebugString 调试字符串的输出(包括发布模式构建的程序),可以说这是个神器,大家可以在微软 MSDN 库上搜索下载。接下来我们运行 DebugView 调试信息捕捉工具。
首先我们打开 DebugView:
我们可以看到随时有程序在输出调试信息,接下来我们运行(按Ctrl+F5 运行,不要启动调试器)刚才的程序,输入 222,看看 DebugView 有什么反应:
这个调试信息查看工具显示了正确的调试信息。注意!如果挂载了 Visual Studio 的调试器,这个工具就会失效。
使用这个工具可以捕获发布版程序的输出,我们一般可以用来跟踪测试产品的运行状态。以趁早发现我们程序中存在的问题,避免发布后出现的容易被检测到的BUG。