zoukankan      html  css  js  c++  java
  • 一步步使用Code::Blocks进行设置断点调试程序

    一、调试之前要做的工作

    首先,我们要确保Code::Blocks的配置正确,调试工作才能进行得更顺利

    为此,我们需要生成调试符号。调试符号可以让调试器知道代码的哪一行正在执行,这样你就可以知道程序运行到哪里了。

    为确保调试符号设置正确,请在Code::Blocks中选择项目 | 编译选项(Project | Build Options),会看到这样一个对话框:

    你需要确保调试(Debug)目标里的生成调试符号(Produce debugging symbols)选项被勾选上。

    还需要在编译 | 选择目标 | 调试(Build | Select Target |Debug)中,确保调试(Debug)作为项目的目标被选中:

    以上操作确保了目标是对项目进行调试,调试器将使用调试符号来编译你的程序。

    二、设置断点

    调试器的价值在于,它能让我们看到程序正在做的事情——哪些代码正在执行,以及变量的值是多少。

    为此,我们在程序的某个地方设置断点,然后在调试器下运行该程序。调试器将执行程序,直到到达设置了断点的代码行。此时,编译器便可以让你查看程序,或者一步步地执行程序,检查代码的每一行是如何影响你的变量的。

    首先我们来看一段用来计算特定数额资金的利率(interest rate)、年利息(compounded annually)的程序:

    #include <iostream>
    using namespace std;
    double computeInterest (double base_val, double rate, int years)
    {
        double final_multiplier;
        for ( int i = 0; i < years; i++ )
        {
            final_multiplier *= (1 + rate);
        }
        return base_val * final_multiplier;
    }
    
    int main ()
    {
        double base_val;
        double rate;
        int years;
        cout << "Enter a base value: ";
        cin >> base_val;
        cout << "Enter an interest rate: ";
        cin >> rate;
        cout << "Enter the number of years to compound: ";
        cin >> years;
    
        cout << "After " << years << " you will have " << computeInterest( base_val, rate, years ) << " money" << endl;
    }

     看了运行结果,很明显,出现了错误,这时我们将设置断点开始进行调试。

    1,先在main函数开始的地方,设置一个断点。这样就可以查看整个程序的执行过程了。

    (1)将光标移到 double base_val 这一行

    (2)选择 调试 | 设置断点 (Debug | Toggle Breakpoint)或者按下F5。这会在该代码行旁边的侧边栏中设置一个小红点,表明这一行有一个断点:

    (3)可以使用设置断点命令或者单击小红点用来设置或取消设置该断点。

    (4)开始运行程序。选择调试 | 开始(Debug | Start)或者按下F8。

    这样程序将正常执行,直到遇到断点。

    现在我们应该看到了打开的调试器,它看起来应该是这样的:

    首先要注意的是小圆点下面的三角形,它表示接下来要执行的代码行。它跟小红点之间相隔若干行。

    它之所以没有紧挨着小红点,是因为变量的声明不产生任何的机器代码,因此,尽管断点看起来是在15行,但实际上它在第18行。

    (5)这时应该还有一个监视(Watches)窗口打开了,如下图:

    我已经展开了监视窗口的两个子项:局部变量(Local variables)和函数参数(Function Arguments)。

    监视窗口会显示出所有当前可用的变量,包括局部变量和函数参数,以及这些变量的值。

    注意:这里看起来像乱码的原因是因为我们还没有对它们进行初始化,这也是接下来的几行程序所要做的事情。

    (6)为了执行接下来的几行代码,我们需要告诉调试器向下执行下一行(F7)

    所谓向下执行一行,就会执行当前的代码行,也就是三角形所标识的那一行。

    一旦走到下一行,程序就会执行cout语句,输出一条信息到屏幕中,要求你输入一个值。

    如果你尝试输入一个值但没有任何效果——因为程序还在调试器的控制之下。

    再次按下F7后,程序会等待用户输入,因为这时候cin函数还未返回——cin函数需要在返回前得到用户的输入。

    重复这一过程,分别输入0.1给利率,输入1给年数。

    现在,断点到达了这一行代码:

    再次确认输入是否正确。我们可以通过监视窗口来检查局部变量的值:

    注意:rate的值不是0.1,是因为0.10000...1中最末尾的1只是浮点数的一种怪异的表达方式(浮点数并不是精确的),它实在太小了,对大多数程序来说不会造成很大影响。

    (7)现在我们确定一切都没问题,来调查一下computerInterest函数中会发生什么,单步执行(Step into)

     单步执行会进入当前行的函数里面去执行,而不像下一行命令,只是执行函数然后显示给你最终的结果。

    现在我们就单步进入computerInterest函数之中:

    (8)从结果中我们可以看出函数的参数部分一切正常,但变量i 和 final_multiplier 值不对劲。

    为此,使用下一行命令(F7),执行循环语句,由于它与一些初始化操作相关联,我们看看会发生什么。

    (9)从中我们可以看出,final_multiplier没有正确初始化。而且,接下来要执行的语句将要用到final_multiplier:

        final_multiplier *= (1 + rate);

    这条语句的意思是,将final_multiplier乘以(1+rate),再把结果重新赋值给final_multiplier。但是我们看到final_multiplier并没有被初始化,因此这个乘法的结果也将会是一个莫名其妙的值。

    (10)如何修复bug?

    我们需要在声明final_multiplier变量的语句中,把它也初始化。在这个例子中,它应该被初始化1。

    double final_multiplier=1;

    (11)修复bug后的运行结果为:

    三、总结

    通过以上一个简单的程序案例,使我掌握了调试一个程序的基本流程,和分析bug的过程,为以后自己独立寻找bug,解决bug提供了实用的技能。

    我不光可以写bug,还能Debug!

  • 相关阅读:
    oracle ORA01001,请求资源正忙或无效
    Flex[Embed(source='assets/error.png')]无法解析用于转换的代码错误
    正在覆盖未标记为 override 的函数
    LINUX umask详解
    C++服务器学习路线
    计算kappa系数
    umask的含义及设置
    Seurat的各种数据成员访问
    GAN训练判别器和生成器时的顺序与detach
    autograd.grad 学习
  • 原文地址:https://www.cnblogs.com/OctoptusLian/p/7202225.html
Copyright © 2011-2022 走看看