mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-16 22:42:21 +08:00
685aa027bd
复查
167 lines
5.6 KiB
Markdown
167 lines
5.6 KiB
Markdown
使用GDB命令行调试器调试C/C++程序
|
||
============================================================
|
||
没有调试器的情况下编写程序时最糟糕的状况是什么?编译时,跪着祈祷不要出错?用生命在运行可执行程序(blood offering不知道怎么翻译好...)?或者在每一行代码间添加printf("test")语句来定位错误点?如你所知,编写程序时不使用调试器的话是不利的。幸好,linux下调试还是很方便的。大多数人使用的IDE都集成了调试器,但linxu著名的调试器是命令行形式的C/C++调试器GDB。然而,与其他命令行工具一致,DGB需要一定的练习才能完全掌握。这里,我会告诉你GDB的基本情况及使用方法。
|
||
|
||
###安装GDB###
|
||
|
||
大多数的发行版仓库中都有GDB
|
||
|
||
Debian 或 Ubuntu
|
||
|
||
$ sudo apt-get install gdb
|
||
|
||
Arch Linux
|
||
|
||
$ sudo pacman -S gdb
|
||
|
||
Fedora,CentOS 或 RHEL:
|
||
|
||
$sudo yum install gdb
|
||
|
||
如果在仓库中找不到的话,可以从官网中下载[official page][1]
|
||
|
||
###示例代码###
|
||
|
||
当学习GDB时,最好有一份代码,动手试验。下列代码是我编写的简单例子,它可以很好的体现GDB的特性。将它拷贝下来并且进行实验。这是最好的方法。
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
|
||
int main(int argc, char **argv)
|
||
{
|
||
int i;
|
||
int a=0, b=0, c=0;
|
||
double d;
|
||
for (i=0; i<100; i++)
|
||
{
|
||
a++;
|
||
if (i>97)
|
||
d = i / 2.0;
|
||
b++;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
###GDB的使用###
|
||
|
||
首先最重要的,你需要使用编译器的 “-g“选项来编译程序,这样可执行程序才能通过GDB来运行。通过下列语句开始调试:
|
||
|
||
$ gdb -tui [executable's name]
|
||
|
||
使用”-tui“选项可以将代码显示在一个窗口内(被称为”文本接口”),在这个窗口内可以使用光标来操控,同时在下面输入GDB shell命令。
|
||
|
||
![](https://farm3.staticflickr.com/2947/15397534362_ac0b5692c8_z.jpg)
|
||
|
||
现在我们可以在程序的任何地方设置断点。你可以通过下列命令来为当前源文件的某一行设置断点。
|
||
|
||
break [line number]
|
||
|
||
或者为一个特定的函数设置断点:
|
||
|
||
break [function name]
|
||
|
||
甚至可以设置条件断点
|
||
|
||
break [line number] if [condition]
|
||
|
||
例如,在我们的示例代码中,可以设置如下:
|
||
|
||
break 11 if i > 97
|
||
|
||
![](https://farm3.staticflickr.com/2948/15374839066_8c7c0eb8a4_o.png)
|
||
|
||
这样,程序循环97次之后停留在“a++”语句上。这样是非常方便的,避免了我们需要手动循环97次。
|
||
|
||
最后但也是很重要的是,我们可以设置一个“观察断点”,当这个被观察的变量发生变化时,程序会被停止。
|
||
|
||
watch [variable]
|
||
|
||
可以设置如下:
|
||
|
||
watch d
|
||
|
||
当d的值发生变化时程序会停止运行(例如,当i>97为真时)。
|
||
当设置后断点后,使用"run"命令开始运行程序,或按如下所示:
|
||
|
||
r [程序的输入参数(如果有的话)]
|
||
|
||
gdb中,大多数的单词都可以简写为一个字母。
|
||
不出意外,程序会停留在11行。这里,我们可以做些有趣的事情。下列命令:
|
||
|
||
bt
|
||
|
||
回溯功能可以让我们知道程序如何到达这条语句的。
|
||
|
||
![](https://farm3.staticflickr.com/2943/15211202760_1e77a3bb2e_z.jpg)
|
||
|
||
info locals
|
||
|
||
这条语句会显示所有的局部变量以及它们的值(你可以看到,我没有为d设置初始值,所以它现在的值是任意值)。
|
||
|
||
当然
|
||
|
||
![](https://farm4.staticflickr.com/3843/15374838916_8b65e4e3c7_z.jpg)
|
||
|
||
p [variable]
|
||
|
||
这可以显示特定变量的值,但是还有更好的:
|
||
|
||
ptype [variable]
|
||
|
||
|
||
可以显示变量的类型。所以这里可以确定d是double型。
|
||
|
||
![](https://farm4.staticflickr.com/3881/15397534242_3cb6163252_o.jpg)
|
||
|
||
既然已经到这一步了,我么不妨这么做:
|
||
|
||
set var [variable] = [new value]
|
||
|
||
这样会覆盖变量的值。不过需要注意,你不能创建一个新的变量或改变变量的类型。我们可以这样做:
|
||
|
||
set var a = 0
|
||
|
||
![](https://farm3.staticflickr.com/2949/15211357497_d28963a9eb_o.png)
|
||
|
||
如其他优秀的调试器一样,我们可以单步调试:
|
||
|
||
step
|
||
|
||
使用如上命令,运行到下一条语句,也可以进入到一个函数里面。或者使用:
|
||
|
||
next
|
||
|
||
这可以直接下一条语句,并且不进入子函数内部。
|
||
|
||
![](https://farm4.staticflickr.com/3927/15397863215_fb2f5912ac_o.jpg)
|
||
|
||
结束测试后,删除断点:
|
||
|
||
delete [line number]
|
||
|
||
从当前断点继续运行程序:
|
||
|
||
continue
|
||
|
||
退出GDB:
|
||
|
||
quit
|
||
|
||
总结,有了GDB,编译时不用祈祷上帝了,运行时不用血祭(?)了,再也不用printf(“test“)了。当然,这里所讲的并不完整,而且GDB的功能远不止这些。所以我强烈建议你自己更加深入的学习它。我现在感兴趣的是将GDB整合到Vim中。同时,这里有一个[备忘录][2]记录了GDB所有的命令行,以供查阅。
|
||
|
||
你对GDB有什么看法?你会将它与图形调试器对比吗,它有什么优势呢?对于将GDB集成到Vim有什么看法呢?将你的想法写到评论里。
|
||
|
||
--------------------------------------------------------------------------------
|
||
|
||
via: http://xmodulo.com/gdb-command-line-debugger.html
|
||
|
||
作者:[Adrien Brochard][a]
|
||
译者:[SPccman](https://github.com/SPccman)
|
||
校对:[校对者ID](https://github.com/校对者ID)
|
||
|
||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创翻译,[Linux中国](http://linux.cn/) 荣誉推出
|
||
|
||
[a]:http://xmodulo.com/author/adrien
|
||
[1]:https://www.gnu.org/software/gdb/
|
||
[2]:http://users.ece.utexas.edu/~adnan/gdb-refcard.pdf
|