• 使用GDB命令行调试器调试C/C++程序


    没有调试器的情况下编写程序时最糟糕的状况是什么?编译时跪着祈祷不要出错?用血祭召唤恶魔帮你运行程序?或者在每一行代码间添加printf("test")语句来定位错误点?如你所知,编写程序时不使用调试器的话是不方便的。幸好,linux下调试还是很方便的。大多数人使用的IDE都集成了调试器,但 linux 最著名的调试器是命令行形式的C/C++调试器GDB。然而,与其他命令行工具一致,DGB需要一定的练习才能完全掌握。这里,我会告诉你GDB的基本情况及使用方法。

    安装GDB

    大多数的发行版仓库中都有GDB

    Debian 或 Ubuntu

    1. $ sudo apt-get install gdb

    Arch Linux

    1. $ sudo pacman -S gdb

    Fedora,CentOS 或 RHEL:

    1. $sudo yum install gdb

    如果在仓库中找不到的话,可以从官网中下载

    示例代码

    当学习GDB时,最好有一份代码,动手试验。下列代码是我编写的简单例子,它可以很好的体现GDB的特性。将它拷贝下来并且进行实验——这是最好的方法。

    1. #include <stdio.h>
    2. #include <stdlib.h>
    3.  
    4. int main(int argc, char **argv)
    5. {
    6. int i;
    7. int a=0, b=0, c=0;
    8. double d;
    9. for (i=0; i<100; i++)
    10. {
    11. a++;
    12. if (i>97)
    13. d = i / 2.0;
    14. b++;
    15. }
    16. return 0;
    17. }

    GDB的使用

    首先最重要的,你需要使用编译器的 “-g“选项来编译程序,这样可执行程序才能通过GDB来运行。通过下列语句开始调试:

    1. $ gdb -tui [可执行程序名]

    使用”-tui“选项可以将代码显示在一个漂亮的交互式窗口内(所以被称为“文本用户界面 TUI”),在这个窗口内可以使用光标来操控,同时在下面的GDB shell中输入命令。

    现在我们可以在程序的任何地方设置断点。你可以通过下列命令来为当前源文件的某一行设置断点。

    1. break [行号]

    或者为一个特定的函数设置断点:

    1. break [函数名]

    甚至可以设置条件断点

    1. break [行号] if [条件]

    例如,在我们的示例代码中,可以设置如下:

    1. break 11 if i > 97

    这样,程序循环97次之后停留在“a++”语句上。这样是非常方便的,避免了我们需要手动循环97次。

    最后但也是很重要的是,我们可以设置一个“观察断点”,当这个被观察的变量发生变化时,程序会被停止。

    1. watch [变量]

    这里我们可以设置如下:

    1. watch d

    当d的值发生变化时程序会停止运行(例如,当i>97为真时)。

    当设置断点后,使用"run"命令开始运行程序,或按如下所示:

    1. r [程序的输入参数(如果有的话)]

    gdb中,大多数的命令单词都可以简写为一个字母。

    不出意外,程序会停留在11行。这里,我们可以做些有趣的事情。下列命令:

    1. bt

    回溯功能(backtrace)可以让我们知道程序如何到达这条语句的。

    1. info locals

    这条语句会显示所有的局部变量以及它们的值(你可以看到,我没有为d设置初始值,所以它现在的值是任意值)。

    当然:

    1. p [变量]

    这个命令可以显示特定变量的值,而更进一步:

    1. ptype [变量]

    可以显示变量的类型。所以这里可以确定d是double型。

    既然已经到这一步了,我么不妨这么做:

    1. set var [变量] = [新的值]

    这样会覆盖变量的值。不过需要注意,你不能创建一个新的变量或改变变量的类型。我们可以这样做:

    1. set var a = 0

    如其他优秀的调试器一样,我们可以单步调试:

    1. step

    使用如上命令,运行到下一条语句,有可能进入到一个函数里面。或者使用:

    1. next

    这可以直接运行下一条语句,而不进入子函数内部。

    结束测试后,删除断点:

    1. delete [行号]

    从当前断点继续运行程序:

    1. continue

    退出GDB:

    1. quit

    总之,有了GDB,编译时不用祈祷上帝了,运行时不用血祭了,再也不用printf(“test“)了。当然,这里所讲的并不完整,而且GDB的功能远远不止于此。所以我强烈建议你自己更加深入的学习它。我现在感兴趣的是将GDB整合到Vim中。同时,这里有一个备忘录记录了GDB所有的命令行,以供查阅。

    引用:

    https://linux.cn/article-4302-1.html

  • 相关阅读:
    配置 dovecat 的 PAM
    通过CVE-2017-17215学习路由器漏洞分析,从入坑到放弃
    路由器漏洞复现分析第二弹:CNVD-2018-01084
    路由器漏洞复现分析第三弹:DVRF INTRO题目分析
    重新认识被人遗忘的HTTP头注入
    [转载]DLL劫持生成器 源码开放(纯WINDOWS SDK)+ 实例分析
    HOOK大法实现不修改程序代码给程序添加功能
    老树开新花:DLL劫持漏洞新玩法
    分享三个USB抓包软件---Bus Hound,USBlyzer 和-USBTrace
    网易云音乐PC客户端加密API逆向解析
  • 原文地址:https://www.cnblogs.com/jiftle/p/10193446.html
Copyright © 2020-2023  润新知