精华内容
下载资源
问答
  • C语言程序调试方法简介.doc
    2021-05-19 20:06:23

    C语言程序调试方法简介

    错误的类型

    (1) 语法错误:拼写错,括号不匹配,漏写分号,…。对于查出的错误(Error)必须排除,否则程序无法运行;而警告(Warning)则应根据情况处理,否则可能产生运算误差等。

    (2) 逻辑错误:编译无误,有时也有执行结果,但结果不符合题意。例如scanf的参数中漏写地址符,if语句、for语句花括号位置错误等,都会导致此类错误。

    (3) 运行错误:运行结果错误也可能是由于输入数据错误、类型不匹配等造成的。例如用户没有按照scanf规定的格式输入数据就会造成此类错误。

    查错的方法

    (1)静态检查:人工检查,程序的结构、各函数间的调用关系,拼写检查。

    (2)编译程序:由C编译系统对程序进行查错,根据错误提示找出错误的位置并改正。注意提示的出错行未必是真正出错的行,常需要向上面的行寻找;而且系统指出的错误类型也未必是真正的错误,需要分析,不能停留在字面上。代码中有一个错误时,可能产生一大批编译错误,应从上到下逐一改正,修改一两个后再次编译。

    (3)排除语法错误后,运行程序,输入数据,得出结果,还应对结果进行分析,看是否符合要求。要准备一些测试数据,有意识地检查结果的正误。

    (4)若运行结果错误,通常由于程序中存在逻辑错误,应对照流程图检查算法逻辑。

    (5)对于怀疑出错的地方,添加一些printf函数输出某些变量的值,以找到出错的程序段,缩小查错范围。

    程序的测试

    通过输入预先准备的数据,找出程序不能正常运行的情况和原因。

    例如解一元二次方程的程序,就应输入以下几组数据进行测试,并验算结果:

    1,2,1 —— 两个相等的实数解

    2,4,2 —— 与上面一组系数的解相同

    4,2,1 —— 提示无实数解,或求出复数解

    1.5,-6.83,2.12 —— 看程序对于实数系数能否正确地输入及运算

    0,1,2 —— 一元一次方和,一个解

    0,0,5 —— 无解

    0,0,0 —— 任意实数解

    VC++中常用的程序调试工具

    VC++6.0提供单步运行、断点跟踪等工具,帮助程序员查错。

    使用功能键或相应的菜单项进行操作:

    F9:在当前光标所在行设置断点(Breakpoint,再次使用则取消当前行已有断点),见图1。

    图1 断点

    F5(Go):调试状态运行程序,程序执行到有断点的地方停下。见图2。

    此时左下角的变量窗口(Variables)显示当前的变量值。

    如果在右下角的察看窗口(Watch)输入变量名,则可监控该变量值的变化。

    图2 运行到断点处,变量窗口,察看窗口

    F10(Step Over)或F11(Step Into):单步执行程序。

    F11和F10的区别:如果当前执行语句是函数调用,则F11会进入被调用函数里面。

    CTRL+F10:运行到光标所在行。

    SHIFT+F11跳出当前所在函数。

    Shift + F5(Stop Debugging):停止调止状态。

    在调试状态,系统会出现Debug(调试)工具箱。其中包括了上面提到的功能的工具按钮。

    单步运行调试的基本步骤:

    (1) 保存C或C++文件。

    (2) 根据断点调试找到错误处。

    (3) 采用F10或F11单步调试找到精确的错误处。可先用F10,确定函数输入输出是否与预想的一致;如不相符,则用F11进入函数体一步一步调试。

    (4) 调试过程中,需要监视程序中变量值的变化。VC++6.0的Variables和Watch窗口,就用来设置监视变量。在调试过程中,鼠标轻轻放在变量上(不用单击)也会显示该变量的值。

    展开阅读全文

    更多相关内容
  • c语言程序调试方法有哪些

    千次阅读 2021-05-22 18:45:59
    c语言程序调试方法有哪些发布时间:2020-04-25 17:23:53来源:亿速云阅读:153作者:小新今天小编给大家分享的是c语言程序调试方法有哪些,相信很多人都不太了解,为了让大家更加了解c语言程序调试方法,所以...

    c语言程序的调试方法有哪些

    发布时间:2020-04-25 17:23:53

    来源:亿速云

    阅读:153

    作者:小新

    今天小编给大家分享的是c语言程序的调试方法有哪些,相信很多人都不太了解,为了让大家更加了解c语言程序的调试方法,所以给大家总结了以下内容,一起往下看吧。一定会有所收获的哦。

    c语言程序的调试方法

    所谓程序调试是指对程序的查错和排错。

    调试程序一般应经过以下几个步骤:

    一、先进行人工检查,即静态检查。

    在写好一个程序以后,不要匆匆忙忙上机,而应对程序进行人工检查。这一步十分重要,它能发现程序设计人员由于疏忽而造成的

    多数错误。这一步往往容易被人忽视,总希望把一切都推给计算机去做,但这样会多占用机器时间,作为一个程序人员应当养成严谨的作风,每一步都要严格把关,不把问题留给后面的工序。

    为了更有效地进行人工检查,所编的程序应力求做到以下几点:①应当采用结构化程序方法编程,以增加可读性;

    ②尽可能多加注释,以帮助理解每段程序的作用;

    ③在编写复杂的程序时不要将全部语句都写在main函数中,而要多利用函数,用一个函数来实现一个单独的功能。各函数之间除用

    参数传递数据外,尽量少出现耦合关系,这样便于分别检查和处理。

    二、在人工检查无误后,再上机调试。

    通过上机发现错误称为动态检查。在编译时会给出语法错误的信息,调试时可以根据提示信息具体找出程序中出错之处并改正。应

    当注意的是有时提示出错的地方并不是真正出错的位置,如果在提示出错的行找不到错误的话应当到上一行再找。有时提示出错的类型并非绝对准确,由于出错的情况繁多且各种错误互有关联,因此要善于分析,找出真正的错误,而不要只从字面意义上找出错信息,钻牛角尖。

    如果系统提示的出错信息很多,应当从上到下逐一改正。有时显示出一大片出错信息往往使人感到问题严重,无从下手。其实可能

    只有一二个错误。例如,对使用的变量未定义,编译时就会对所有含该变量的语句发出出错信息。这时只要加上一个变量定义,就所有错误都消除了。

    三、在改正语法错误(包括“错误(error)”和“警告(warning)”)后,程序经过连接(link)就得到可执行的目标程序。运行

    程序,输入程序所需数据,就可得到运行结果。应当对运行结果作分析,看它是否符合要求。有的初学者看到运行结果就认为没问题了,不作认真分析,这是危险的。

    有时,数据比较复杂,难以立即判断结果是否正确。可以事先考虑好一批“试验数据”,输入这些数据可以很容易判断结果正确与

    否。例如解方程ax2+bx+c=0,输入a、b、c的值分别为1、-2、1时,根x的值是1。这是容易判断的,若根不等于1,程序显然有错。

    但是,用“试验数据”时,程序运行结果正确,还不能保证程序完全正确。因为有可能输入另一组数据时运行结果不对。例如,用公式求根x的值,当a≠0和b2-4ac>0时,能得出正确结果,当a=0或b2-4ac<0时,就得不到正确结果(假设程序中未对a=0作防御处理以及未作复数处理)。因此应当把程序可能遇到的各种情况都一一试到。例如,if语句有两个分支,有可能程序在经过其中一个分支时结果正确,而经过另一个分支时结果不对。必须考虑周全。事实上,当程序复杂时很难把所有的可能情况全部都试到,选择典型的临界数据作试验即可。

    四、运行结果不对,大多属于逻辑错误。对这类错误往往需要仔细检查和分析才能发现。可以采用以下办法:1.将程序与流程图仔细对照,如果流程图是正确的,程序写错了,是很容易发现的。例如,复合语句忘记写花括弧,只要一对照流

    程图就能很快发现。

    2.如果实在找不到错误,可以采用“分段检查”的方法。在程序不同的位置设几个printf函数语句,输出有关变量的值,逐段往下

    检查。直到找到在某一段中数据不对为止。这时就已经把错误局限在这一段中了。不断减小“查错区”,就能发现错误所在。

    3.也可以用“条件编译”命令进行程序调试(在程序调试阶段,若干printf函数语句就要进行编译并执行。当调试完毕,这些语句

    不用再编译了,也不再被执行了)。这种方法可以不必一一去掉printf函数语句,以提高效率。

    4.如果在程序中没有发现问题,就要检查流程图有无错误,即算法有无问题。如有则改正之,接着修改程序。

    5.有的系统还提供debug(调试)工具,跟踪程序并给出相应信息,使用更为方便,请查阅有关手册。

    总之,程序调试是一项细致深入的工作,需要下功夫,动脑子,善于积累经验。在程序调试过程中往往反映出一个人的水平,经验

    和态度。希望大家给以足够的重视。上机调试程序的目的决不是为了“验证程序的正确”,而是“掌握调试的方法和技术”,要学会自己找问题,这样慢慢自己就会写出错误较少的实用程序。

    关于c语言程序的调试方法有哪些就分享到这里了,当然并不止以上和大家分析的办法,不过小编可以保证其准确性是绝对没问题的。希望以上内容可以对大家有一定的参考价值,可以学以致用。如果喜欢本篇文章,不妨把它分享出去让更多的人看到。

    展开全文
  • 文章目录一、前言二、调试的概念以及调试器的选择调试器(Debugger)1) Remote Debugger2) WinDbg3) LLDB4) GDB三、设置断点,开始调试断点的真正含义继续执行程序删除断点代替暂停语句四、查看和修改变量的值更加...

    一、前言

    C语言常见错误有三种:

    1. 语法错误:指程序中含有不符合语法规定的语句
    2. 逻辑错误:指程序中没有语法错误,可以通过编译、连接生成可执行程序,但程序运行的结果与预期不相符的错误。
    3. 运行时错误:一种是由于考虑不周或输入错误导致程序异常(Exception),比如数组越界访问,除数为零,堆栈溢出等等。另一种是由于程序设计思路的错误导致程序异常或难以得到预期的效果。

    对于语法错误,编译器可以帮助我们解决;对于逻辑错误和运行时错误,编译器可以通过,但是在实际使用过程中得不到预期的结果或者运行崩溃,我们就需要通过调试来解决。

    二、调试的概念以及调试器的选择

    所谓调试(Dubug),就是跟踪程序的运行过程,从而发现程序的逻辑错误(思路错误),或者隐藏的缺陷(Bug)。

    在调试的过程中,我们可以监控程序的每一个细节,包括变量的值、函数的调用过程、内存中数据、线程的调度等,从而发现隐藏的错误或者低效的代码。

    我敢保证,每个人都会遇到逻辑错误,而且会经常遇到,初学者更是错的离谱,所以,必须掌握调试技能,没有选择的余地,没有学会调试就是没有学会编程!

    在编写代码的过程中,相信大家肯定遇到过这样的情况:代码能够编译通过,没有语法错误,但是运行结果却不对,反复检查了很多遍,依然不知道哪里出了问题。这个时候,就需要调试程序了。

    所谓调试(Debug),就是让代码一步一步慢慢执行,跟踪程序的运行过程。比如,可以让程序停在某个地方,查看当前所有变量的值,或者内存中的数据;也可以让程序一次只执行一条或者几条语句,看看程序到底执行了哪些代码。

    在调试的过程中,我们可以监控程序的每一个细节,包括变量的值、函数的调用过程、内存中数据、线程的调度等,从而发现隐藏的错误或者低效的代码。

    编译器可以发现程序的语法错误,调试可以发现程序的逻辑错误。所谓逻辑错误,是指代码思路或者设计上的缺陷。

    对于初学者来说,学习调试也可以增加编程的功力,它能让我们更加了解自己自己的程序,比如变量是什么时候赋值的、内存是什么时候分配的,从而弥补学习的纰漏。

    调试是每个程序员必须掌握的技能,没有选择的余地!

    调试器(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 将是一个不错的选择。

    考虑到大部分读者使用Windows,本教程以 VS2010 为例讲解调试技巧,也即使用 Remote Debugger 调试器。当然你也可以使用 VS2012、VS2013、VS2015 等,它们之间的差别很小。

    三、设置断点,开始调试

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

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

    img

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

    img

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

    img

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

    • 调用堆栈可以看到当前函数的调用关系。
    • 断点窗口可以看到当前设置的所有断点。
    • 即时窗口可以让我们临时运行一段代码,后续我们会重点讲解。
    • 输出窗口和我们之前看到的没有,用来显示程序的运行过程,给出错误信息和警告信息。
    • 自动窗口会显示当前代码行和上一代码行中所使用到的变量。
    • 局部变量窗口会显示当前函数中的所有局部变量。
    • 线程模块窗口暂时无需理会。

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

    img

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

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

    img

    断点的真正含义

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

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

    继续执行程序

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

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

    img

    删除断点

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

    img

    代替暂停语句

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

    四、查看和修改变量的值

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

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

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

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

    img

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

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

    img

    更加快捷的方式

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

    img

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

    img

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

    添加监视

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

    img

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

    img

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

    五、单步调试(逐语句调试和逐过程调试)

    在实际开发中,常常会出现这样的情况,我们可以大致把出现问题的代码锁定在一定范围内,但无法确定到底是哪条语句出现了问题,该怎么办呢?按照前面的思路,必须要在所有代码行前面设置断点,让代码一个断点一个断点地执行。

    这种方案确实可行,但很麻烦,也不专业,这节我们就来介绍一种更加便捷的调试技巧——单步调试。所谓单步调试,就是让代码一步一步地执行。

    下面的代码用来求一个等差数列的和,我们以该代码来演示单步调试:

    #include <stdio.h>
    int main(){
        int start, space, length, i, thisNum;
        long total = 0;
       
        printf("请输入等差数列的首项值:");
        scanf("%d", &start);
        printf("请输入等差数列的公差:");
        scanf("%d", &space);
        printf("请输入等差数列的项数:");
        scanf("%d", &length);
       
        for(i=0; i<length; i++){
            thisNum = start + space * i;
            if( length-i > 1 ){
                printf("%d + ", thisNum);
            }else{
                printf("%d", thisNum);
            }
            total += thisNum;
        }
        printf(" = %ld\n", total);
       
        return 0;
    }
    

    在第6行设置一个断点,然后点击“逐过程调试”按钮,或者按F10键,程序就会运行到下一行并暂停:

    img

    再次点击“逐过程”按钮或按F11键,就会执行第7行代码,要求用户输入数据。用户输入结束后,黄色箭头就会指向第8行,并暂停程序。如此重复执行上面的操作,就可以让程序一条语句一条语句地执行,以观察程序的内部细节,这就称为单步调试。

    逐过程调试(F10)和逐语句调试(F11)

    刚才我们在第6行设置了断点,按下“逐过程”按钮或F10键,程序就会执行 printf(),并暂停在下一行代码处。

    printf() 是一个函数,它本身由多条语句构成,如果你希望进入 printf() 函数内部,查看整个函数的执行过程,可以点击“逐语句”按钮,或者按F11键,这样程序就会进入 printf() 所在的源文件,如下图所示:

    img

    当需要跳出函数时,可以点击“跳出”按钮,或者按Shift+F11键,就会返回刚才的代码。

    逐过程(F10)逐语句(F11)都可以用来进行单步调试,但是它们有所区别:

    • 逐过程(F10)在遇到函数时,会把函数从整体上看做一条语句,不会进入函数内部;
    • 逐语句(F11)在遇到函数时,认为函数由多条语句构成,会进入函数内部。

    逐语句(F10)不仅可以进入库函数的内部,还可以进入自定义函数内部。在实际的调试过程中,两者结合可以发挥更大的威力。

    断点 + 查看/修改变量 + 逐过程调试 + 逐语句调试,这足以解决绝大多数逻辑问题,到此,初学者就已经学到了调试的基本技能。

    修改代码运行位置

    在VS中,调试器还允许我们直接跳过一段代码,不去执行它们。将下面的代码复制到源文件:

    #include <stdio.h>
    int main(){
        printf("111\n");
        printf("222\n");
        printf("333\n");
        printf("444\n");
        printf("555\n");
        printf("666\n");
        return 0;
    }
    

    在第3行设置断点,开始单步调试。假设我们不希望执行4~6行的代码,那么当程序执行到第4行时,可以将鼠标移动到黄色箭头处,直接向下拖动到第7行,如下图所示:

    img

    程序执行完成后,在控制台上会输出:
    111
    555
    666

    注意:随意修改程序运行位置是非常危险的行为,假设我们定义了一个指针,在第N行代码中让它指向了一个数组,如果我们在修改程序运行位置的时候跳过了第N行代码,并且后面也使用到了该指针,那么就极有可能导致程序崩溃。

    六、即时窗口的使用

    “即时窗口”是VS提供的一项非常强大的功能,在调试模式下,我们可以在即时窗口中输入C语言代码并立即运行,如下图所示:

    img

    在即时窗口中可以使用代码中的变量,可以输出变量或表达式的值(无需使用printf()函数),也可以修改变量的值。

    即时窗口本质上是一个命令解释器,它负责解释我们输入的代码,再由VS中的对应模块执行,最后将输出结果呈现到即时窗口。

    需要注意的是,在即时窗口中不能定义新的变量,因为程序运行时 Windows 已经为它分配好了只够刚好使用的内存,定义变量是需要额外分配内存的,所以调试器不允许在程序运行的过程中定义变量,因为这可能会导致不可预知的后果。

    调用函数

    在即时窗口中除了可以使用代码中的变量,也可以调用代码中的函数。将下面的代码复制到源文件中:

    int plus(int x, int y){
        return x + y;
    }
    int main(){
        return 0;
    }
    

    在第6行设置断点,并在即时窗口中输入plus(5, 6),如下图所示:

    img

    七、查看、修改运行时的内存

    在 Visual Studio 的调试过程中,有时候我们可能需要对程序的运行内存进行查看,修改该怎么办?Visual Studio 也为我们提供了很好的解决方案。那就是使用 Visual Studio 自带的内存查看窗口。

    首先我们通过内存窗口查看变量的值,我们启动 Visual Studio,创建一个工程,输入如下代码:

    #include <stdio.h>
    int main()
    {
        int testNumber = 5678;
        printf("testNumber 的内存地址为 0x00%x \n", &testNumber);
        //输出内存地址
        //TODO:在这里插入断点
        return 0;
    }
    

    我们在第七行设置好断点,然后按 F5 启动调试,等待触发断点。触发断点后,我们发现,IDE中并没有显示内存窗口(默认设置下),这时,我们点击菜单 -> 调试(D) -> 窗口 (W) -> 内存 (M) -> 内存1(1),就可以调出内存窗口了,如图:

    img

    我们看到,内存窗口里面显示着一大堆乱七八糟的数据,这里面的这些就是我们内存中的数据啦,我们可以通过变量 testNumber 的内存地址跳转到器对应的内存空间,我们看到 testNumber 的地址为 0x0018f830 (在不同的进程是不一样的),我们把这个地址输入到我们的内存窗口的地址栏。如图:

    img

    我们看到,尽管我们已经输入了正确地地址,但是我们还是没有看到正确的数据显示,其实原因非常简单,我们来回顾一下 C 语言的一些入门知识:我们知道,在我们的源代码中,我们的 testNumber 变量被定义为 int 整形,我们再想想 int 整形在内存中占几个字节?没错,是4个字节。所以我们应该以四字节的形式格式化这些内存数据,这是我们在内存窗口中单击我们的鼠标右键,在弹出的菜单中选择“4字节整数(4)”,然后就能正确地显示相关的数据了,如图:

    img

    没错,查看内存就是这么的简单。接下来我们就来查看与修改浮点数的内存数据,我们看下面这段代码:

    #include <stdio.h>
    int main()
    {
        double pi = 3.141592653589;
        printf("pi 的内存地址为 %x \n", &pi);
        //输出内存地址
        //TODO:在这里插入断点
        return 0;
    }
    

    同样的,我们在第7行设置断点,按F5启动调试,等待断点被触发:

    img

    这时我们看到的内存地址是这样的,与我们在内存窗口看到的不同,我们需要将其补齐,在我们现阶段编写的小程序中,显示的内存地址基本上都是六位的,我们在前面加上 “0x00”,将其补到八位(内存窗口上的地址栏里有几位就补到几位)。然后我们将其输入到内存窗口上的地址栏。

    img

    我们发现,现在显示的数据依然是错误的,因为查看器现在还是在使用我们之前设置的 4位整形格式 格式化我们的内存数据呢,我们知道,我们的 double 属于64位浮点数,所以我们在查看窗口点击鼠标右键,在弹出的菜单中选择“64位浮点(6)”,然后我们就能看到它正确地输出数据了。

    img

    我们注意到,在我们设置的变量pi的值的基础上,内存窗口里显示的数据多了几个0,这些0所代表的就是 double 型数据的最高精度。接下来我们尝试在内存窗口修改变量 pi 的值,为其补齐精度。现在我们用鼠标右键点击我们的pi的内存数据,在弹出的菜单中选择编辑值(E),在显示的输入框中我们输入 3.1415926535897931,按回车即可保存。我们看看效果:

    img

    怎么样,内存的查看与修改是不是很简单呢?其实我们只要记住下面的几个对应关系,常用的数值数据类型的内存查看与修改都不在话下:

    类型名变量类型内存查看窗口中应选择的数据格式
    short16 位整形2 字节整数
    int32 位整形4 字节整数
    long32 位整形4 字节整数
    long long64 位整形8 字节整数
    float32 位(4字节)单精度浮点数32 位浮点
    double64 位(8字节)双精度浮点数64 位浮点

    内存新手常用的内存窗口操作就到这里了,其实关于内存操作还有很多高级用法,不过那不在我们初级调试教程的范围之内。我们在修改内存的时候要注意安全,防止随意修改导致的程序崩溃,甚至是无法结束进程!

    八、有条件断点的设置

    在此之前,我们已经了解了无条件断点、跟踪点这两种断点,事实上在 Visual Studio 中还有几种常用的断点,在本节我们将一一为大家介绍。

    大家有没有碰到这样的情况,在一个循环体中设置断点,假设有一千次循环,我们想在第五百次循环中设置断点,这该怎么办?反正设置断点不断按 F5 继续运行四百九十九次是不可能的。那该怎么办呢?其实我们的断点是可以设置各种各样的条件的,对于上述情况,我们可以对断点的命中次数做一个限制。

    我们首先在 Visual Studio 中创建一个工程,并且输入如下代码:

    #include <stdio.h>
    int main(){
        for ( int i=1 ; i <= 1000 ; i++ ) {
            //TODO:插入计次断点
            printf("我真行!\n");
        }
    }
    

    首先,我们在第4行插入断点,分析代码,我们不难得出它会输出 1000 行“我真行!”,那么我们思考一下,在不修改代码的情况下,如何才能让他输出 1499 行“我真行!”呢,其实很简单,我们只要在i 等于500的时候暂停程序,再将变量 i 的值修改为 1 即可,思路很简单,接下来我们就来实现这个命中条件的限制吧。

    首先我们用鼠标右键单击第4行的断点图标,在弹出的菜单中选择 命中次数(H) ,接下来会弹出如下图的一个对话框,我们在中间的选择框中选择 “中断,条件是命中次数等于”,我们在右边的编辑框输入 500。

    img

    我们点击确定,断点就设置到位了,接下来我们按 F5 运行调试。

    img

    我们看到,在输出四百九十九行“我真行!”后,程序进入了中断状态,这是我们注意到自动窗口中的变量 i 的值为 500,接下来我们把这个 i 的值改为 1,点击 继续© 继续程序的运行,这样程序就再输出了一千行“我真行!”,然后退出。没错,命中次数限制的使用就是这么简单。

    我们再次用鼠标右键单击第4行的断点图标,在弹出的菜单中选择 命中次数(H) ,大家如果有兴趣的话,可以试试中间的选择框中其他的条件选项,使用方法基本一致,这里不再赘述。

    接下来我们来了解一下断点触发条件的使用,在 Visual Studio 的调试器中,我们可以对断点设置断点触发条件,这个条件可以引用我们程序中的变量,比如我们程序中有两个变量 a、b ,我们的命中条件可以是 a == b 、 a >= b 、 a != b 甚至是 (a - b)(a2 - b) > 0 这样的复杂条件。

    下面我们就来试试在 Visual Studio 中插入条件断点吧。我们首先创建一个工程,输入如下代码:

    #include <stdio.h>
    #include <stdlib.h>
    #include <time.h> //time函数所在头文件
    int main()
    {
        int a, b;
        int randNumber;
        srand((unsigned)time(NULL));
        //设置随机数种子,以产生不同随机数
        for (int i = 0; i<50; i++)
        {
            a = rand() % 7; //产生0-6的随机数
            b = rand() % 7; //产生0-6的随机数
            //TODO:在这里插入条件断点: a == b
        }
        return 0;
    }
    

    我们让程序运行过程中 a 等于 b 的时候触发断点,首先,我们在第十四行插入断点,然后我们鼠标右键单击左侧的断点图标,在弹出的菜单中选择条件©,IDE会弹出如下对话框,我们在条件输入框中输入 a==b ,然后在下面选择 为 true ,然后点击确定即可。

    九、assert断言函数

    在我们的实际开发过程之中,常常会出现一些隐藏得很深的BUG,或者是一些概率性发生的BUG,通常这些BUG在我们调试的过程中不会出现很明显的问题,但是如果我们将其发布,在用户的各种运行环境下,这些程序可能就会露出马脚了。那么,如何让我们的程序更明显的暴露出问题呢?这种情况下,我们一般都会使用 assert 断言函数,这是C语言标准库提供的一个函数,也就是说,它的使用与操作系统平台,调试器种类无关。我们只要学会了它的使用,便可一次使用,处处开花。

    接下来我们来了解一下 assert 函数的用法,这个函数在 assert.h 头文件中被定义,在微软的 cl 编译器中它的原型是这样的:

    #define assert(_Expression) (void)( (!!(_Expression)) || (_wassert(_CRT_WIDE(#_Expression), _CRT_WIDE(__FILE__), __LINE__), 0) )
    

    我们看到 assert 在 cl 编译器中被包装成了一个宏,实际调用的函数是 _wassert ,不过在一些编译器中,assert 就是一个函数。为了避免这些编译器差异带来的麻烦,我们就统一将 assert 当成一个函数使用。

    我们来了解一下 assert 函数的用法和运行机制,assert 函数的用法很简单,我们只要传入一个表达式即可,它会计算我们传入的表达式的结果,如果为真,则不会有任何操作,但是如果我们传入的表达式的计算结果为假,它就会像 stderr (标准错误输出)打印一条错误信息,然后调用 abort 函数直接终止程序的运行。

    现在我们在 Visual Studio 中创建一个工程,输入下面这段非常简单的代码:

    #include <stdio.h>
    #include <assert.h>
    int main()
    {
        printf("assert 函数测试:");
        assert(true); //表达式为真
        assert(1 >= 2); //表达式为假
        return 0;
    }
    

    我们按 F5 启动调试:

    img

    我们看到,我们的输出窗口打印出了断言失败的信息,并且 Visual Studio 弹出了一个对话框询问我们是否继续执行。但是如果我们不绑定调试器,构建发布版程序,按 Ctrl + F5 直接运行呢?是的,这个 assert 语句就无效了,原因其实很简单,我们看看 assert.h 头文件中的这部分代码:

    #ifdef NDEBUG
    #define assert(_Expression)     ((void)0)
    #else  /* NDEBUG */
    

    我们看到,只要我们定义了 NDEBUG 宏,assert 就会失效,而 Visual Studio 的默认的发布版程序编译参数中定义了 NDEBUG 宏,所以我们不用额外定义,但是在其他编译器中,我们在发布程序的时候就必须在包含 assert.h 头文件前定义 NDEBUG 宏,避免 assert 生效,否则总是让用户看到“程序已经停止运行,正在寻找解决方案 . . .”的 Windows 系统对话框可就不妙了。

    下面我们来了解一下 assert 的常用情境以及一些注意事项。举个C语言文件操作的例子:

    #include <stdio.h>
    #include <assert.h>
    #include <stdlib.h>
    int main(void)
    {
        FILE *fpoint;
        fpoint = fopen("存在的文件.txt", "w");
        //以可读写的方式打开一个文件
        //如果文件不存在就自动创建一个同名文件
        assert(fpoint);                        
        //(第一次断言)所以这里断言成功
        fclose(fpoint);
        fpoint = fopen("不存在的文件.txt", "r");
        //以只读的方式打开一个文件,如果不存在就打开文件失败
        assert(fpoint);
        //(第二次断言)所以断言失败
        printf("文件打开成功");
        //程序永远都执行不到这里来
        fclose(fpoint);                          
        return 0;
    }
    

    阅读代码,我们可以知道,如果我们错误的设置了读写权限,或者没有考虑 Windows 7(或更高版本) 的 UAC(用户账户权限控制) 问题的话,我们的程序出现的问题就很容易在我们的测试中暴露出来。事实上,上面的代码也不是一定能通过第一次断言的,我们把构建生成的(Debug 模式的)可执行文件复制到我们的系统盘的 Program Files 文件夹再执行,那么 Win7 及以上的操作系统的UAC都会导致程序无法通过第一次断言的。所以在这种情况下我们就需要在必要的情况申请管理员权限避免程序BUG的发生。

    在我们的实际使用过程中,我们需要注意一些使用 assert 的问题。首先我们看看下面的这个断言语句:

    //...
    assert( c1 /*条件1*/ && c2 /*条件2*/ );
    //...
    

    我们思考一下:如果我们的程序在这里断言失败了,我们如何知道是 c1 断言失败还是 c2 断言失败呢,答案是:没有办法。在这里我们应该遵循使用 assert 函数的第一个原则:每次断言只能检验一个条件,所以上面的代码应该改成这样:

    //...
    assert(c1 /*条件1*/);
    assert(c2 /*条件2*/);
    //...
    

    这样,一旦出现问题,我们就可以通过行号知道是哪个条件断言失败了。

    接下来我们来思考一个问题:我们可以使用 assert 语句代替我们的条件过滤语句么?我们来举个例子,我们写了一个视频格式转换器,我们本应输出标准MP4格式的视频,但是由于编码解码库的问题,我们在转换过程中出现了错误,像这种情况下,我们能否用 assert 代替掉我们的 if 语句呢(假设这里忽略了 Visual Studio 发布版程序编译过程中自动定义的 NDEBUG 宏)?很明显,这样做是不可以的。这是为什么呢?

    现在我们来区分一下程序中的错误与异常,所谓错误,是代码编写途中的缺陷导致的,是程序运行中可以用 if 语句检测出来的。而异常在我们的 C 语言中,一般也可以使用 if 语句判断,并进行对应的处理。而 assert 是用来判断我们程序中的代码级别的错误的。像用户输入错误,调用其他函数库错误,这些问题我们都是可以用 if 语句检测处理的。另一方面,使用 if 语句的程序对用户更友好。

    下面我们通过代码展示使用 assert 的另外一个注意意事项,我们新建一个工程,输入如下代码:

    #include <stdio.h>
    #include <assert.h>
    int main(void)
    {
        int i = 0;
        for ( ; ; )
        {
            assert(i++ <= 100);
            printf("我是第%d行\n",i);
        }
        return 0;
    }
    

    我们按 F5 运行调试器,我们会看到这样的情景:

    img

    这是正常的,我们按 Shift + F5 终止调试,接下来,我们切换一下编译模式到发布模式:

    img

    接下来我们按 Ctrl+F5 不绑定调试器直接运行:

    img

    我们看到了一个完全不相同的运行结果,这是为什么呢?其实原因很简单,我们注意这段代码:

    assert(i++ <= 100);

    我们的条件表达式为 i++ <= 100,这个表达式会更改我们的运行环境(变量i的值),在发布版程序中,所有的 assert 语句都会失效,那么这条语句也就被忽略了,但是我们可以把它改为 i++ ; assert(i <= 100); ,这样程序就能正常运行了。所以请记住:不要使用会改变环境的语句作为断言函数的参数,这可能导致实际运行中出现问题。

    最后,我们再来探讨一下,什么时候应该用 assert 语句?一个健壮的程序,都会有30%~50%的错误处理代码,几乎用不上 assert 断言函数,我们应该将 assert 用到那些极少发生的问题下,比如Object* pObject = new Object,返回空指针,这一般都是指针内存分配出错导致的,不是我们可以控制的。这时即使你使用了容错语句,后面的代码也不一定能够正常运行,所以我们也就只能停止运行报错了。

    九、调试信息的输出

    上一节,我们讲解了 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("你好,C语言中文网。"));
        OutputDebugString(_T("大家好才是真的好。"));
        //也可以:OutputDebugStringA("大家好才是真的好。");
        //也可以:OutputDebugStringW(L"大家好才是真的好。");
        //使用自动字符宏 TEXT 或者 _T 可以自动判断是否使用宽字符
        system("pause"); //使程序暂停一下
        return 0;
    }
    

    在程序执行 system(“pause”); 暂停的时候我们来观察一下我们的,调试输出窗口:

    img

    怎么样,是不是输出了“你好,C语言中文网。大家好才是真的好。”呢?这个函数与 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 开始调试:

    img

    我们看到了输出的结果,怎么样,大家是不是觉得这样调用这个函数很麻烦?为解决此问题,这里C语言中文网为大家提供了一个更好的解决方案,我们可以自己写一个前端函数,然后保存到头文件中(编译生成 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 的输出窗口:

    img

    我们可以看到,它正确地输出了 “用户输入的数是 666” ,这就说明我们的代码执行成功了,上面的代码大家可以随意修改我们上面提供的代码以符合自己的实际需要。但是请注意不要让缓冲区溢出,否则会造成不可估计的后果。

    我们除了在调试器中可以看到调试字符串的输出,我们还可以借助 Sysinternals 软件公司研发的一个相当高级的工具 —— DebugView 调试信息捕捉工具,这个工具可以在随时随地捕捉 OutputDebugString 调试字符串的输出(包括发布模式构建的程序),可以说这是个神器,大家可以在微软 MSDN 库上搜索下载。接下来我们运行 DebugView 调试信息捕捉工具。

    首先我们打开 DebugView:

    img

    我们可以看到随时有程序在输出调试信息,接下来我们运行(按Ctrl+F5 运行,不要启动调试器)刚才的程序,输入 222,看看 DebugView 有什么反应:

    img

    这个调试信息查看工具显示了正确的调试信息。注意!如果挂载了 Visual Studio 的调试器,这个工具就会失效。

    使用这个工具可以捕获发布版程序的输出,我们一般可以用来跟踪测试产品的运行状态。以趁早发现我们程序中存在的问题,避免发布后出现的容易被检测到的BUG。

    十、VS调试的总结以及技巧

    我们已经对 Visual Studio 的调试有了一些了解,在这一节,我们将向大家展示我们在调试过程中可以使用的一些实用技巧。使大家的调试操作更加轻松愉快。

    首先我们再了解一下Visual Studio 中,Release构建模式和Debug 构建模式的区别。我们在img可以切换构建模式。Release构建模式下构建的程序为发行版,而Debug构建模式下构建的程序为调试版。在 Visual Studio 中调试模式还会定义两个宏 _DEBUG 和 DEBUG,后文我们将介绍它们的一些妙用。在 Visual Studio 中,如果我们要更改编译参数的话,可以点击菜单 -> 项目§ -> <项目名>属性§,我们在弹出的页面左侧选择配置属性即可对编译参数进行修改。

    接下来,我们来了解一下调试标记。不知道大家有没有遇到这样的情况,我们需要在调试的时候额外运行一段代码,但是实际发布的时候却不需要这段代码呢。那该怎么办,绝大多数数的初学者会选择使用注释,即在发布的时候将无用的测试代码注释掉。但是这样很麻烦,下面我们就为大家介绍一种全新的方法——使用调试标记。事实上这种方法我们在前面使用过,但是没有详细讲解。

    这种方法借助了预处理指令,方法很简单,我们首先定义一个宏作为处于调试状态的标记,后面的代码我们用 #ifdef 和 #endif 预处理指令检测宏是否被定义,然后由编译器判断是否编译其中的代码。这么做的好处就是可以减少发布程序的体积,另一方面可以提高发布程序的运行效率。下面是一段示范代码:

    #include <stdio.h>
    #define _DEBUGNOW
    int main(){
        #ifdef _DEBUGNOW
        printf("正在为调试做准备...");
        #endif // _DEBUGNOW
        printf("程序正在运行...");
        return 0;
    }
    

    当我们要发布上面的这个程序的时候,我们只要将 #define _DEBUGNOW 注释掉即可,无需进行任何额外的操作。怎么样?是不是很方便呢?善用调试标记可以大大地提高我们的调试效率,但是有一点记住,调试标记名不要过于简单,否则可能和程序中的变量/常量产生冲突。

    在我们的调试过程中,我们常常会需要在不触发断点的情况下输出一些数值,这时我们一般会这么做:

    printf("%d\n",Value/*要输出的数值*/);
    

    但是像这种代码写多了我们可能会对它产生一种厌恶之情,这是我们的预处理器又可以派上用场了,我们可以定义一个宏解决这个问题,定义宏的代码如下:

    #define _PUTINT(NUM) printf("%d\n",NUM)
    

    然后我们只要这样调用我们的宏:

    _PUTINT(45/*要输出的数值*/);
    

    怎么样?是不是很方便呢?那如果我们想让这个宏在我们发布程序的时候失效呢,我们该怎么做?其实很简单,我们依然可以使用预处理指令完成这项操作,下面我们来看一套完整的代码(同时使用调试标记和宏):

    #include <stdio.h>
    #include <stdlib.h>
    #define _PAUSE() system("pause");
    #if (defined DEBUG) || (defined _DEBUG) //检测构建模式是否为调试模式
    //如果构建模式为调试模式,这里定义几个宏
    #define _DEBUGNOW
    #define _PUTSIL(NUM) printf("%d\n",NUM) //输出整数
    #define _PUTFD(NUM) printf("%f\n",NUM) //输出浮点数
    #else
    //如果构建模式为发布模式,自动忽略这些宏的存在
    #define _PUTSIL(NUM) ((void)0)
    #define _PUTFD(NUM) ((void)0)
    #endif
    int main(){
        #ifdef _DEBUGNOW 
        printf("正在为调试做准备...\n");
        #endif // _DEBUGNOW
        printf("程序正在运行...\n");
        _PUTSIL(12666);
        _PUTFD(3.1415926535898);
        printf("程序运行完毕...\n");
        _PAUSE(); // 暂停程序
        return 0;
    }
    

    我们看到,宏的定义和函数差不多,有的简单的函数可以直接用宏实现,这样做带来的好处就是,我们可以不用频繁的判断是否处于调试状态,一次定义,一直有效。

    为了方便我们的调试(检查)操作以及日后的团队合作,我们在编写函数的时候应该为其加上 Visual Studio 的智能提示,方法比较简单,我们只要在函数定义前面加上提示注释即可(格式自由),Visual Studio 便会自动分析我们的代码并加入其智能提示列表,下面我们举一个定义函数设置智能提示的例子:

    //
    //  函数:  Convert2Jpeg(wchar_t*, wchar_t, int)
    //
    //  目的:    转换图片到 Jpeg 格式
    //
    //  orgiPath : 源文件路径
    //  destPath : 目标路径
    //  quality  : 图像质量
    //
    bool Convert2Jpeg(wchar_t* orgiPath, wchar_t* destPath, int quality){
        return true;
    }
    

    我们在自动完成将或者鼠标移动到编辑器内函数名上,就有了智能提示。
    img

    这种智能提示不但可以减轻我们编写代码时的负担,在调试情境下也可以增加我们检查代码的效率。如此一来,岂不美哉!

    接下来我们为大家总结了一些调试的经验以及调试思路:
    当我们运行我们编写的程序发现运行结果与我们预想的不同的时候,我们可以先用即时窗口,使用一些比较简单的数据来测试我们的各个函数运行结果是否符合我们的预期,如果都符合的话,我们可以使用程序中产生的一些比较复杂的数据来进一步测试我们的各个函数,直至找到可能导致错误的函数。

    找到可能导致错误的函数之后,我们就可以使用逐语句调试来一步步跟踪运行程序了,渐渐的的我们就可以缩小范围直至定位错误(无关代码可以考虑暂时注释掉),在这期间,我们要仔细观察程序运行过程中各个数据的变化情况,观察的仔细与否直接与我们能否找到错误直接挂钩。

    如果上一步运行的数据一直是正常的,我们就可以排除这个函数的嫌疑了(减少对他的调试次数)。此时,我们就应该考虑问题是否出现在之前的函数上了,可能因为偶然性,我们第一次测试函数的时候并没有发现其错误,导致范围锁定产生偏差,此时我们需要再次耐心的对所有未排除嫌疑的进行调试,直至再次找到出错的函数。再重复上一步,直至找到错误。

    可以看到,调试其实是一项比较复杂的活,需要大量的操作,所以在我们编写代码的时候要万分谨慎!因为很多时候,BUG都是因为我们的粗心大意导致的笔误引起的!

    十一、DEV C++调试教程

    上面的教程是使用VS来演示的,下面我们也简单介绍一下使用DEV C++来进行调试的过程。

    调试步骤如下

    一、打开调试工具

    1、点击菜单栏中的工具[T]

    img

    2、选择编译选项[C]

    img

    3、更改调试信息
    点击【代码生成/优化】,将【连接器】中【产生调试信息】一项的No改为Yes,然后点击下方的确定即可。

    img

    二、通过鼠标查看变量

    1、点击菜单栏中的工具[T],选择环境选项[v]

    img

    2、更改信息
    找到【浏览Debug变量】,选中【查看鼠标指向的变量】,然后点击下方的确定即可。

    img

    使用方法:当开始调试以后,把鼠标划过代码中想查看的变量就可以在左边调试一栏中查询该变量的当前值。

    img

    三、调试方法

    1、打开一个.c或.cpp文件,编译它,一方面,要确保这个代码能够编译通过,另一方面,在每次调试前都是需要先编译的。

    2、设置“断点”
    在要设置断点的那行代码开头处的数字上单击一下即可,可以设置多个断点,也可以只设置一个断点。如果设置了多个断点,程序会在断点与断点之间进行调试。如果只有一个断点,程序会从设置断点处开始,随着每次点击,一步一步进行下去,直到程序结束。

    img

    3、点击“调试”,开始进入调试
    点击菜单栏中的【√】或者下方的【调试】都可以。

    img

    4、如果程序需要输入,那么点击进入后,就会跳出黑框。输入数据,按“回车”。蓝色一行表示当前程序运行的位置。

    img

    5、调试开始后,可以点击调试一栏中的‘下一步’,让程序运行到想要的位置。

    ref:http://c.biancheng.net/view/vip_2085.html
    https://www.jianshu.com/p/1602264dadf2

    展开全文
  • C语言预定义跟踪调试

    2020-10-25 18:27:45
    标准C语言预处理要求定义某些对象宏,每个预定义宏的名称一两个下划线字符开头和结尾,这些预定义宏不能被取消定义(#undef)或由编程人员重新定义。下面预定义宏表,被我抄了下来。 LINE :当前程序行的行号,表示...

    标准C语言预处理要求定义某些对象宏,每个预定义宏的名称一两个下划线字符开头和结尾,这些预定义宏不能被取消定义(#undef)或由编程人员重新定义。下面预定义宏表,被我抄了下来。

    LINE :当前程序行的行号,表示为十进制整型常量
    FILE :当前源文件名,表示字符串型常量
    DATE :转换的日历日期,表示为Mmm dd yyyy 形式的字符串常量,Mmm是由asctime产生的。
    TIME :转换的时间,表示"hh:mm:ss"形式的字符串型常量,是有asctime产生的。(asctime貌似是指的一个函数)
    STDC :编辑器为ISO兼容实现时位十进制整型常量
    STDC_VERSION :如何实现复合C89整部1,则这个宏的值为19940SL;如果实现符合C99,则这个宏的值为199901L;否则数值是未定义
    STDC_EOBTED :(C99)实现为宿主实现时为1,实现为独立实现为0
    STDC_IEC_559 :(C99)浮点数实现复合IBC 60559标准时定义为1,否者数值是未定义
    STDC_IEC_559_COMPLEX: (C99)复数运算实现复合IBC 60559标准时定义为1,否者数值是未定义
    STDC_ISO_10646 :(C99)定义为长整型常量,yyyymmL表示wchar_t值复合ISO 10646标准及其指定年月的修订补充,否则数值未定义
    C++中还定义了 __cplusplus

    C语言中的__FILE__、__LINE__和__DATE__等都在头文件#include<stdio.h>中

    其他说明:

    如果编译器不是标准的,则可能仅支持以上宏名中的几个,或根本不支持。记住编译程序也许还提供其它预定义的宏名。
    LINEFILE 宏指示,#line指令可以改变它的值,简单的讲,编译时,它们包含程序的当前行数和文件名。
    STDC 宏指令的意义是编译时定义的。一般来讲,如果__STDC__已经定义,编译器将仅接受不包含任何非标准扩展的标准C/C++代码。如果实现是标准的,则宏__STDC__含有十进制常量1。如果它含有任何其它数,则实现是非标准的。
    __cplusplus 与标准c++一致的编译器把它定义为一个包含至少6为的数值。与标准c++不一致的编译器将使用具有5位或更少的数值。

    常用宏的具体例子如下:

    C语言中的__LINE__用以指示本行语句在源文件中的位置信息,举例如下:

    C语言__FILE__、__LINE__等预定义跟踪调试
    5

    6

    7

    还可以通过语句#line来重新设定__LINE__的值,举例如下:

    C语言__FILE__、__LINE__等预定义跟踪调试
    编译执行后输出结果为:

    203

    204

    205

    C语言中的__FILE__用以指示本行语句所在源文件的文件。

    例如,test.c文件内容如下:

    C语言__FILE__、__LINE__等预定义跟踪调试
    在gcc编译生成a.out,执行后输出结果为:

    test.c

    C语言中__DATE__和__TIME__表示时间和日期

    C语言__FILE__、__LINE__等预定义跟踪调试
    结果:

    Aug 26 2020
    23:18:22

    __STDC__是预定义宏。当它被定义后,编译器将按照ansic标准来编译你的c程序。

    __cplusplus用来定义是否是C++编译器

    C语言__FILE__、__LINE__等预定义跟踪调试
    输出结果为:

    C

    但是如果在前面定义_cplusplus

    C语言__FILE__、__LINE__等预定义跟踪调试
    那么输出就是:

    C++

    C
    另外gcc还支持__func__,和__FUNCTION__,它指示所在的函数,但是这个关键字不被windows下的vc6.0支持,举例如下

    C语言__FILE__、__LINE__等预定义跟踪调试
    其编译后输出结果为

    main

    main

    注意: “#line”、 “LINE”、 “FILE" 及 “func" 都是大小写敏感的。

    展开全文
  • c语言程序调试(实验)使用VC++6.0调试程序 调试程序 调试运行 单步跟踪 单步跳进跟踪 观察自动变量 观察其他变量 停止调试 1:打开jiecheng项目(双击jiecheng.dsw文件) 2:build该项目,确定程序可以运行 3:调试运行...
  • C语言工程模块跟踪实例: 模块选配 转载于:https://www.cnblogs.com/CyJack/p/10881321.html
  • c语言程序调试方法整理

    千次阅读 2021-05-18 10:31:22
    c语言程序调试方法所谓程序调试是指对程序的查错和排错。调试程序一般应经过以下几个步骤:一、先进行人工检查,即静态检查。在写好一个程序以后,不要匆匆忙忙上机,而应对程序进行人工检查。这一步十分重要,它...
  • 所谓程序调试是指对程序的查错和排错。 调试程序一般应经过以下几个步骤: 一、先进行人工检查,即静态检查。 在写好一个程序以后,不要匆匆忙忙上机,而应对程序进行人工检查。这一步十分重要,它能发现程序...
  • 写在前面 本文以将gdb调试为主,顺便提一下strace和valgrind。 正文 ... gdb可以直接调试程序,也可以将程序和core文件一块调试,也可以调试服务程序,只需要指定对应服务程序的进程id就行了...
  • Linux下C语言程序调试

    千次阅读 2018-10-04 16:07:11
    在Linux中调试C语言程序1.编译时添加调试信息2.使用调试命令进行调试gdb调试器语法gdb模式下的命令[^1]**readelf命令**ELF文件类型ELF文件作用ELF文件总体组成选项 1.编译时添加调试信息 使用gcc -g -o [生成...
  • 关于C语言跟踪调试方法 C语言跟踪调试技术有助于发现C程序中的问题逻辑错误具体用法有4步 设置断点当跟踪运行时断点可以使程序在断点处暂停 跟踪运行与普通运行程序不同它用于调试程序时运行快捷键为F5 单步执行按...
  • C语言程序调试常用技巧和常见错误分析C语言程序调试常用技巧和常见错误分析摘要:在C语言程序的开发过程中经常会伴随着多次对C语言程序进行修改调试的过程。本文讨论了C语言程序调试中常用技巧,分析了调试中常见...
  • c语言程序设计上半部分习题课动态跟踪调试PPT课件.pptx
  • c语言程序设计上半部分习题课动态跟踪调试PPT学习教案.pptx
  • 高级语言程序设计 上半部分习题课 ;课程内容;重点;难点;程序数据结构算法 ;要点1运算符及表达式;要点2C语言的基本数据类型;要点3条件if语句;要点4循环语句;要点5函数;要点6基本输入输出语句;第二章习题讲解;1.指出...
  • 设了断点以后如何走程序 进入调试: 1. 点此按钮 2. 按F5 3. Build?Start Debug?Go 黄色右箭头表示当前就要执行的代码行 当前执行中的上下文语境函数 自动给出的当前语境下的变量值返回值观察窗口 Debug工具栏 ...
  • C语言简单的单步调试

    千次阅读 2021-04-15 18:20:19
    C语言中的单步跟踪调试 单步调试是指程序开发中,为了找到程序的bug,通常采用的一种调试手段,一步一步跟踪程序执行的流程,根据变量的值,找到错误的原因。 下面以一个简单的小程序为例,这里的程序必须是在项目下...
  • C语言调试

    2021-05-23 03:40:55
    所谓调试(Debug),就是让代码一步一步慢慢执行,跟踪程序的运行过程。比如,可以让程序停在某个地方,查看当前所有变量的值,或者内存中的数据;也可以让程序一次只执行一条或者几条语句,看看程序到底执行了哪...
  • 第三 C语言程序调试

    2019-12-25 17:29:32
    C语言学习栏目目录 目录 4.1 int类型介绍 4.2 char类型介绍 4.3 float、double类型介绍 4.4 小结及其他数据类型简单介绍 4.5 类型大小 首先,还是先看程序 // 有错误的程序 #include <stdio.h> int...
  • 当你在程序运行的时候用 Ctrl + C,程序的运行能够终止,而GDB能展示它的当前地址、堆栈跟踪信息之类的内容。  但是它是怎么办到的呢?  但是它们怎么不工作呢?  开始,让我们先研究它怎样才会不工作。它不能...
  • 文章目录20201学期《C语言程序设计B》平时自主学习D13428.C对于跳转后的数组可见性的...请单击此处下载文件D13428.C,然后对程序进行跟踪调试,要求不增加或删除行,测试时输入的数据为6 1 3 2 9 -222。 程序的功能是:
  • 所谓调试(Dubug),就是跟踪程序的运行过程,从而发现程序的逻辑错误(思路错误),或者隐藏的缺陷(Bug)。 在调试的过程中,我们可以监控程序的每一个细节,包括变量的值、函数的调用过程、内存中数据、线程的...
  • C语言程序设计实验报告(五)专业 计算机科学与技术 班级 卓越工程师班 日期 2011年12月09日 实验组别 第一组 成绩 第五次实验 编译预处理实验 指导教师 李开 学生姓名 邱金源 学号 U201114493实验名称 编译预处理实验...
  • Linux下C语言调试

    2021-05-15 21:09:08
    1、C语言预处理定义的一些宏可以帮助我们进行调试好预处理定义的宏宏说明__LINE__代表当前行号的十进制常数__FILE__代表当前文件名的字符串__DATE__代表当前日期,Mmm dd yyyy格式的字符串__TIME__代表当前时间,hh:...
  • C语言代码跟踪调试

    千次阅读 2014-04-11 17:19:33
    在linux编程中,当文件数量变的众多之后,使用gdb调试就是一场灾难。因此在程序中加入合理的打印信息, 定位错误出现的文件名,函数名,行号等信息, 能更高效的定位到问题的所在。  下面定义了宏,分别是...
  • C语言中的调试小技巧

    2021-05-19 14:49:41
    经常看到有人介绍一些IDE或者像gdb这样的调试器的很高级的调试功能,也听人说过有些牛人做工程的时候就用printf来调试,不用特殊的调试器。特别是在代码经过编译器一些比较复杂的优化后,会变得“难以辨认”,使用...
  • c语言程序调试方法所谓程序调试是指对程序的查错和排错。调试程序一般应经过以下几个步骤:一、先进行人工检查,即静态检查。在写好一个程序以后,不要匆匆忙忙上机,而应对程序进行人工检查。这一步十分重要,它...
  • C语言gdb调试

    2021-10-12 14:55:49
    修改系统资源限制参数2.core 文件存放的路径3.gdb 调试加上 core 文件五、调试正在运行中的程序1.运行示例程序2.查看进程编号3.gdb 调试加入进程编号六、调试多进程服务程序1.示例程序2.先用 gdb 进去调试3.gdb默认...
  • C语言课件 第十四章 常见错误和程序调试》由会员分享,可在线阅读,更多相关《C语言课件 第十四章 常见错误和程序调试(36页珍藏版)》请在人人文库网上搜索。1、第十四章常见的错误和程序调试,第十四章常见的...

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 27,005
精华内容 10,802
关键字:

c语言程序跟踪调试