前言
在Linux程序开发过程中,语法错误,编译时会报错,很好排查。但如果是开发者的逻辑错误或者处理失误,虽然编译过,程序也可以程序运行,但是功能却没有达到理想的效果,甚至在运行过程中突然崩溃。查找原因时,有时也不好排查。这时候,就可以借助GBD工具来调式程序。
GDB简介
UNIX及UNIX-like下的调试工具。或许,各位比较喜欢那种图形界面方式的,像VC、BCB等IDE的调试,但如果你是在 UNIX平台下做软件,你会发现GDB这个调试工具相比于VC、z的优点是具有修复网络断点以及恢复链接等功能,比BCB的图形化调试器有更强大的功能。所谓“尺有所短,寸有所长”就是这个道理。
上面一段文字是官方版,可能看了也是模棱两可。其实一般只需要知道,gdb是GNU开源组织发布的一个强大的Linux下的程序调试工具,可以用来调试C和C++程序。
然后就是功能,一般来说,GDB主要帮助你完成下面四个方面的功能:
1、启动你的程序,可以按照你的自定义的要求随心所欲的运行程序。
2、可让被调试的程序在你所指定的调置的断点处停住。(断点可以是条件表达式)
3、当程序被停住时,可以检查此时你的程序中所发生的事。
4、你可以改变你的程序,将一个BUG产生的影响修正从而测试其他BUG。
是的,还是调试的意思。
GDB使用
首先需要将你的程序代码编译成二进制,并且在编译时加参数"-g",再用gdb打开即可。例如:
gcc xxx.c -o xxx.exe -g
生成 xxx.exe,再执行
gdb xxx.exe -q
则进入调试,-q 是清除GDB版本等信息,不加也行。
如果是xxx.cpp也就是c++程序,就是g++编译 ,反正编译时后面加参数"-g" 就行了。
调试例子
#include <stdio.h>
int main()
{
int a=10;
for( int i=a-1;i>0;i--)
printf(":%d\n",i);
return 0;
}
这是一个c程序,下面一些调试操作
kch@kch:~$ gcc test.c -o test.exe -g
kch@kch:~$ gdb test.exe -q
Reading symbols from test.exe...done.
(gdb) l
1 #include <stdio.h>
2 int main()
3 {
4 int a=10;
5 for( int i=a;i>0;i--)
6 printf(":%d\n",i);
7 return 0;
8 }
9
(gdb) b 5
Breakpoint 1 at 0x400535: file test.c, line 5.
(gdb) r
Starting program: /home/kch/test.exe
Breakpoint 1, main () at test.c:5
5 for( int i=a;i>0;i--)
(gdb) p a
$1 = 10
(gdb) n
6 printf(":%d\n",i);
(gdb) n
:10
5 for( int i=a;i>0;i--)
(gdb) p a
$2 = 10
(gdb) p i
$3 = 10
(gdb) n
6 printf(":%d\n",i);
(gdb) p i
$4 = 9
(gdb)
GDB常用命令
- 运行指令:
run(简写r): 运行程序,当遇到断点后,程序会在断点处停止运行,等待用户输入下一步命令
continue(简写c) : 继续执行,到下一个断点停止(或运行结束)
next(简写n) : 单步跟踪程序,当遇到函数调用时,也不进入此函数体;
此命令同 step 的主要区别是,step 遇到用户自定义的函数,将步进到函数中去运行,而 next 则直接调用函数,不会进入到函数体内。
step (简写s):单步调试如果有函数调用,则进入函数;与命令n不同,n是不进入调用的函数的
until(简写u):当你厌倦了在一个循环体内单步跟踪时,这个命令可以运行程序直到退出循环体。
until+行号: 运行至某行,不仅仅用来跳出循环finish: 运行程序,直到当前函数完成返回,
并打印函数返回时的堆栈地址和返回值及参数值等信息。
call 函数(参数):调用程序中可见的函数,并传递“参数”,如:call gdb_test(55)quit(简写q) : 退出gdb
- 设置断点:
break n (简写b n):在第n行处设置断点(可以带上代码路径和代码名称: b OAGUPDATE.cpp:578)
b fn1 if a>b:条件断点设置
break func(break缩写为b):在函数func()的入口处设置断点,如:break cb_button
delete 断点号n:删除第n个断点
disable 断点号n:暂停第n个断点
enable 断点号n:开启第n个断点clear 行号n:清除第n行的断点
info b (info breakpoints) :显示当前程序的断点设置情况
delete breakpoints:清除所有断点:
- 查看源代码:
list :简记为 l ,其作用就是列出程序的源代码,默认每次显示10行。
list 行号:将显示当前文件以“行号”为中心的前后10行代码,如:list 12
list 函数名:将显示“函数名”所在函数的源代码,如:list main
list :不带参数,将接着上一次 list 命令的,输出下边的内容。
- 打印表达式
print 表达式(简记p): 其中“表达式”可以是任何当前正在被测试程序的有效表达式,比如当前正在调试C语言的程序,
那么“表达式”可以是任何C语言的有效表达式,包括数字,变量甚至是函数调用。
print a:将显示整数 a 的值
print ++a:将把 a 中的值加1,并显示出来
print name:将显示字符串 name 的值print
gdb_test(22):将以整数22作为参数调用 gdb_test() 函数print
gdb_test(a):将以变量 a 作为参数调用 gdb_test() 函数
display 表达式:在单步运行时将非常有用,使用display命令设置一个表达式后,它将在每次单步进行指令后,
紧接着输出被设置的表达式及值。如: display a
watch 表达式:设置一个监视点,一旦被监视的“表达式”的值改变,gdb将强行终止正在被调试的程序。如: watch a
whatis :查询变量或函数
info function: 查询函数
扩展info locals: 显示当前堆栈页的所有变量
- 查询运行信息
where/bt :当前运行的堆栈列表;
bt backtrace 显示当前调用堆栈
up/down 改变堆栈显示的深度
set args 参数:指定运行时的参数
show args:查看设置好的参数
info program: 来查看程序的是否在运行,进程号,被暂停的原因。
XBB
GDB调试确实很强,但是很多人喜欢山赛方法,printf调试法,顾名思义就是在某些地方加打印信息来调试,简单粗暴,但程序比较繁琐的话,效率就极低,特别费事。而且如果你去面试,人家大多都会问你会不会GDB调试,虽然也不一定要用,要是人家就是要问。虽然,GDB还是要掌握的。