C语言:调试的概念和调试器的选择

所谓调试(Dubug),就是跟踪程序的运行过程,从而发现程序的逻辑错误(思路错误),或者隐藏的缺陷(Bug)。 在调试的过程中,我们可以监控程序的每一个细节,包括变量的值、函数的调用过程、内存中数据、线程的调度等,从而发现隐藏的错误或者低效的代码。 我敢保证,每个人都会遇到逻辑错误,而且会经常遇到,初学者更是错的离谱,所以,必须掌握调试技能,没有选择的余地,没有学会调试就是没有学会编程!

调试器(Debugger)

调试需要借助专业的辅助软件------调试器(Debugger)。现在主流 C/C++调试器有下面几种:

1) Remote Debugger

Remote Debugger 是 VC/VS 自带的调试器,与整个 IDE 无缝衔接,使用非常方便,初学者建议使用该调试器,本教程也以 VS2010 为例讲解调试技巧。

2) WinDbg

大名鼎鼎的 Windows 下的调试器,它的功能甚至超越了 Remote Debugger,它还有一个命令行版本 (cdb.exe),但是这个命令行版本的调试器指令比较复杂,不建议初学者使用。

3) LLDB

XCode 自带的调试器,Mac OS X 下开发必备调试器。

4) GDB

Linux 下使用最多的一款调试器,也有 Windows 的移植版,如果你不使用 VC/VS,GDB 将是一个不错的选择。

默认情况下,程序不会进入调试模式,代码会瞬间从开头执行到末尾。要想观察程序的内部细节,就得让程序在某个地方停下来,我们可以在这个地方设置断点。

所谓断点(BreakPoint),可以理解为障碍物,人遇到障碍物不能行走,程序遇到断点就暂停执行。

上图中,我们希望让程序在第 4 行代码处暂停执行,那么在第 4 行代码左侧的灰色部分单击鼠标即可插入断点。你也可以将光标定位到要暂停的代码行,然后按 F9 键插入断点。也可以在要暂停的位置单击鼠标右键,在弹出菜单中插入断点,如下图所示:

插入断点后,点击上方的"运行"按钮,或者按 F5 键,即可进入调试模式,如下图所示:

可以看到,程序虽然运行了,但并未输出任何数据,这是因为在输出数据之前就暂停了。同时,在 IDE 的上方出现了与调试相关的工具条,下方也多出了几个与调试相关的窗口:

 调用堆栈可以看到当前函数的调用关系。

 断点窗口可以看到当前设置的所有断点。

 即时窗口可以让我们临时运行一段代码,后续我们会重点讲解。

 输出窗口和我们之前看到的没有,用来显示程序的运行过程,给出错误信息和警告信息。

 自动窗口会显示当前代码行和上一代码行中所使用到的变量。

 局部变量窗口会显示当前函数中的所有局部变量。

 线程和模块窗口暂时无需理会。

如果你的 VS 缺少某个窗口,可以通过 VS 上方的"调试"菜单调出,如下图所示:

注意:必须在调试状态下才能看到图中的菜单。

如果你希望关闭某个窗口,可以在窗口标题处单击鼠标右键,在弹出菜单中隐藏,如下图所示:

断点的真正含义

严格来说,调试器遇到断点时会把程序暂时挂起,让程序进入一种特殊的状态------中断状态,这种状态下操作系统不会终止程序的执行,也不会清除与程序相关的元素,比如变量、函数等,它们在内存中的位置不会发生变化。

关键是,处于中断状态下的程序允许用户查看和修改它的运行状态,比如查看和修改变量的值、查看和修改内存中的数据、查看函数调用关系等,这就是调试的奥秘。

继续执行程序

点击"运行"按钮或者按 F5 键即可跳过断点,让程序恢复正常状态,继续执行后面的代码,直到程序结束或者遇到下一个断点。

在调试过程中,按照上面的方法可以设置多个断点,程序在执行过程中每次遇到断点都会暂停,如下图所示:

删除断点

如果不希望程序暂停,可以删除断点。删除断点也很简单,在原有断点处再次单击鼠标即可,也可以将光标定位到要删除断点的代码行,再次按 F9 键,或者在右键菜单中删除,如下图所示

代替暂停语句

在 VS 下,程序运行结束后不会自动暂停(一闪而退),要手动添加暂停语句 system("pause");,如果大家觉得麻烦,也可以在代码最后插入断点,强制程序暂停。

设置了断点,就可以观察程序的运行情况了,其中很重要的一点就是查看相关变量的值,这足以发现大部分逻辑错误。

将下面的代码复制到源文件中:

1. #include <stdio.h>
2. int main(){
3. int value_int, array_int[3];
4. float value_float;
5. char* value_char_pointer;
6. //在这里插入断点
7. value_int = 1048576;
8. value_float = 2.0;
9. value_char_pointer = "Hello World";
10. array_int[0] = 379; array_int[1] = 94;
11. //在这里插入断点
12. return 0;
13. }

在第 7 行和第 12 行插入断点。运行到第一个断点时,在局部变量窗口可以看到各个变量的值:

可以看到,未经初始化的局部变量和数组的值都是垃圾值,是随机的,没有意义。双击变量的值,可以进行修改。

点击"运行"按钮或按 F5 键,程序会运行到下一个断点位置,在局部变量窗口可以看到各个值的变化:

更加快捷的方式

除了在窗口中查看变量,还有一种更加便捷的方法:在调试模式下,把鼠标移动到要查看的变量的上方,即可看他它的值。如下图所示:

如果是数组、指针、结构体等还可以展开,如下图所示:

这种查看变量的方式在实际开发中使用很多。

添加监视

如果你希望长时间观测某个变量,还可以将该变量添加到监视窗口。在要监视的变量处单击鼠标右键,弹出如下菜单:

选择"添加监视",在 VS 下方的监视窗口就可以看到当前变量:

这样,每次变量的值被改变都会反映到该窗口中,无需再将鼠标移动到变量上方查看其值。尤其是当程序稍大时,往往需要同时观测多个变量的值,添加监视的方式就会显得非常方便。

相关推荐
RoadToTheExpert22 分钟前
PHP 5 6 7 8 9 各重要版本开发特性和选择简要说明
开发语言·php
泰山小张只吃荷园37 分钟前
SCAU软件体系结构期末复习-名词解释题
java·开发语言·后端·学习·spring·面试
李老头探索1 小时前
深入理解 Java Set 集合:原理、应用与高频面试题解析
java·开发语言
翔云 OCR API1 小时前
手机号认证接口、C++API核验、实名认证
开发语言·c++
SoraLuna1 小时前
「Mac畅玩鸿蒙与硬件48」UI互动应用篇25 - 简易购物车功能实现
开发语言·macos·ui·华为·harmonyos
爱lv行1 小时前
安装和配置 Apache 及 PHP
开发语言·php·apache
向宇it2 小时前
【从零开始入门unity游戏开发之——unity篇04】unity6基础入门——场景窗口(Scene)和层级窗口(Hierarchy)介绍
开发语言·unity·c#·游戏引擎
橘子海全栈攻城狮2 小时前
【源码+文档+调试讲解】“健康早知道”微信小程序
开发语言·servlet·微信小程序·小程序·notepad++
爱是小小的癌3 小时前
Java-数据结构-顺序表(ArrayList)
java·开发语言·数据结构
我明天再来学Web渗透3 小时前
【2024年-11月-9日-开源社区openEuler实践记录】OpenAMDC:开启智能边缘计算与系统管控的新征程
开发语言·人工智能·架构·开源·边缘计算·copilot·开源软件