linux环境编程之gdb工具的详解
GDB命令的使用
GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具。或许,各位比较喜欢那
种图形界面方式的,像VC、BCB等IDE的调试,但如果你是在UNIX平台下做软件,你
会发现GDB这个调试工具有比VC、BCB的图形化调试器更强大的功能。所谓“寸有所长,
尺有所短”就是这个道理。
一般来说,GDB主要帮忙你完成下面四个方面的功能:
1、启动你的程序,可以按照你的自定义的要求随心所欲的运行程序。
2、可让被调试的程序在你所指定的调置的断点处停住。(断点可以是条件表达式)
3、当程序被停住时,可以检查此时你的程序中所发生的事。
4、动态的改变你程序的执行环境。
从上面看来,GDB和一般的调试工具没有什么两样,基本上也是完成这些功能,不过在细
节上,你会发现GDB这个调试工具的强大,大家可能比较习惯了图形化的调试工具,但有
时候,命令行的调试工具却有着图形化工具所不能完成的功能。
<一>有关gdb的常用命令:
//一下操作都是在启动了gdb之后的操作
1》 List
简写:l 列出源文件中的代码,默认一次列出十行,也可以自己指定列出的行数,用法如:
gdb)l (直接回车,列出文件中的十行代码),可以继续点击回车,重复上一次的命令
(**指定列出的行数)
*列出指定区域(n1到n2之间)的代码:
(gdb) list n1,n2
这样,list可以简写为l,将会显示n1行和n2行之间的代码(使用逗号隔开),如果使用-tui启动gdb,将会在相应的位置显示。如果没有n1和n2参数,那么就会默认显示当前行和之后的10行,再执行又下滚10行。另外,list还可以接函数名。
一般来说在list后面可以跟以下这们的参数:
<linenum> 行号。
<+offset> 当前行号的正偏移量。
<-offset> 当前行号的负偏移量。
<filename:linenum> 哪个文件的哪一行。例如:list ReadFile.cpp:n1,n2(该文件中第n1,n2行)
<function> 函数名。
<filename:function> 哪个文件中的哪个函数。
<*address> 程序运行时的语句在内存中的地址 (????)
*2》启动gdb,并且分屏显示源代码:
$gdb -tui
这样,使用了'-tui'选项,启动可以直接将屏幕分成两个部分,上面显示源代码,比用list方便多了。
这时候使用上下方向键可以查看源代码,想要命令行使用上下键就用[Ctrl]n和[Ctrl]p
3》可以直接使用gdb 可执行程序名,再启动gdb的同时,就将可执行文件加载进去。需要注意的是,载入的app程序必须在编译的时候有gdb调试选项,例如'gcc -g app app.c',注意,如果修改了程序的源代码,但是没有编译,那么在gdb中显示的会是改动后的源代码,但是运行的是改动前的程序,这样会导致跟踪错乱的。
或者,在启动了gdb之后,可以使用gdb 进程号,来启动该可执行文件。
再者,启动程序之后,再用gdb调试:
$gdb <program> <PID>
这里,<program>是程序的可执行文件名,<PID>是要调试程序的PID.如果你的程序是一个服务程序,那么你可以指定这个服务程序运行时的进程ID。gdb会自动attach上去,并调试他。program应该在PATH环境变量中搜索得到。
4》启动gdb之后,可以使用help来查询帮助。
例如:help list,查看list命令的帮助命令
5》再启动gdb程序之后,在进行可执行程序的载入。
*载入指定的程序:
(gdb) file app
这样在gdb中载入想要调试的可执行程序app。如果刚开始运行gdb而不是用gdb app启动的话可以这样载入app程序,当然编译app的时候要加入-g调试选项。
6》运行程序;
(gdb) run
要想运行准备调试的程序,可使用run命令,在它后面可以跟随发给该程序的任何参数,包括标准输入和标准输出说明符(<和> )和shell通配符(*、?、[、])在内。
7》修改发送给程序的参数
(gdb) set args no
这里,假设我使用"r yes"设置程序启动参数为yes,那么这里的set args会设置参数argv[1]为no。
8》*显示缺省的参数列表:使用show args
(gdb) show args
9》*执行下一步:
(gdb) next
这样,执行一行代码,如果是函数也会跳过函数。这个命令可以简化为n.
可以再执行到断点出,之后使用该命令进行一步一步的执行。
(**)也可以命令后面加上具体的数据,表示执行N行,例如:next N
10》*单步进入:
(gdb) step
这样,也会执行一行代码,不过如果遇到函数的话就会进入函数的内部,再一行一行的执行。
11》*执行完当前函数返回到调用它的函数:
(gdb) finish
这里,运行程序,直到当前函数运行完毕返回再停止。例如进入的单步执行如果已经进入了某函数,而想退出该函数返回到它的调用函数中,可使用命令finish.
12》*指定程序直到退出当前循环体:
(gdb) until 或(gdb) u
这里,发现需要把光标停止在循环的头部,然后输入u这样就自动执行全部的循环了。
13》*跳转执行程序到第5行:
(gdb) jump 5
这里,可以简写为"j 5"需要注意的是,跳转到第5行执行完毕之后,如果后面没有断点则继续执行,而并不是停在那里了。
另外,跳转不会改变当前的堆栈内容,所以跳到别的函数中就会有奇怪的现象,因此最好跳转在一个函数内部进行,跳转的参数也可以是程序代码行的地址,函数名等等类似list。
注意(跳转时,最好在同一个函数中进行跳转,否则会出现错误);
14》强制返回d ..当前函数:
(gdb) return
这样,将会忽略当前函数还没有执行完毕的语句,强制返回。return后面可以接一个表达式,表达式的返回值就是函数的返回值。
15》*强制调用函数:
(gdb) call <expr>
这里,<expr>可以是一个函数,这样就会返回函数的返回值,如果函数的返回类型是void那么就不会打印函数的返回值,但是实践发现,函数运行过程中的打印语句还是没有被打印出来。
*强制调用函数2:
(gdb) print <expr>
这里,print和call的功能类似,不同的是,如果函数的返回值是void那么call不会打印返回值,但是print还是会打印出函数的返回值并且存放到历史记录中。
注意:可以使用print var 来打印出某个变量的值
16》设置断点
Break 可以简写: b
使用方法:
Gdb)break 行号,表示在这一行设置一个断点,当程序运行到这一行是们就会进行中断,停止在这一块。
(2)*设置条件断点: break 行号 条件表达式 ,当满足条件时进行设置断点
(gdb) break 46 if testsize==100
这里,如果testsize==100就在46行处断点。
(3)在指定函数的位置设置断点
Break func(函数名)
(4)给某个文件中的某一行设置断点
Break filename:linenum (用法所示)
17》*显示当前gdb断点信息:
(gdb) info breakpoints
这里,可以简写为info break.会显示当前所有的断点,断点号,断点位置等等
18》*删除N号断点:
(gdb) delete N
delete后面什么都不写,表示删除所有的断点。
(**)*清除行N上面的所有断点:
(gdb) clear N
19》*继续运行程序直接运行到下一个断点:
(gdb) continue 简写:c
这里,如果没有断点就一直运行。
20》(gdb) backtrace (显示函数调用以及堆栈列表)
命令产生一张列表,包含着从最近的过程开始的所有有效过程和调用这些过程的参数。当然,这里也会显示出当前运行到了哪里(文件,行号)。
21》*查看当前函数的程序语言:
(gdb) info frame
22》*显示当前的调试源文件: (会显示原文件的信息)
(gdb) info source
这样会显示当前所在的源代码文件信息,例如文件名称,程序语言等。
23》*终止一个正在调试的程序:
(gdb) kill
这里,输入kill就会终止正在调试的程序了。
24》*print显示变量(var)值:
(gdb) print var
这里,print可以简写为p,print 是gdb的一个功能很强的命令,利用它可以显示被调试的语言中任何有效的表达式。表达式除了包含你程序中的变量外,还可以包含函数调用,复杂数据结构和历史等等。
*用16进制显示(var)值:
(gdb) print /x var
这里可以知道,print可以指定显示的格式,这里用'/x'表示16进制的格式。
可以支持的变量显示格式有:
x 按十六进制格式显示变量。
d 按十进制格式显示变量。
u 按十六进制格式显示无符号整型。
o 按八进制格式显示变量。
t 按二进制格式显示变量。
a 按十六进制格式显示变量。
c 按字符格式显示变量。
f 按浮点数格式显示变量。
注意:*如果a是一个数组,10个元素,如果要显示则:
(gdb) print *a@10
这样,会显示10个元素,无论a是double或者是int的都会正确地显示10个元素。
--》*修改运行时候的变量值:
(gdb) print x=4
这里,x=4是C/C++的语法,意为把变量x值改为4,
25》*显示一个变量var的类型:
(gdb) whatis var
*以更详细的方式显示变量var的类型:
(gdb) ptype var
这里,会打印出var的结构定义。
注意:
gcore 该命名用于进行gcore命令来使其主动出Core并退出 。
26》examine命令(简写是x)来查看内存地址中的值。x命令的语法如下所示:
(1)查看内存命令。
x /<n/f/u> <addr>
查看内存变量内容n
:表示显示内存长度(整数)f
:表示显示格式d
:十进制x
:十六进制o
:八进制t
:二进制
u
: 表示显示字节数<addr>
:变量内存地址
- display 表达式
display /i $pc
显示c和汇编同步x /i $pc
程序计数器
(2)x/<n/f/u> <addr>
n、f、u是可选的参数。
n是一个正整数,表示需要显示的内存单元的个数,也就是说从当前地址向后显示几个内存单元的内容,一个内存单元的大小由后面的u定义。
f 表示显示的格式,参见下面。如果地址所指的是字符串,那么格式可以是s,如果地十是指令地址,那么格式可以是i。
u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4个bytes。u参数可以用下面的字符来代替,b表示单字节,h表示双字节,w表示四字 节,g表示八字节。当我们指定了字节长度后,GDB会从指内存定的内存地址开始,读写指定字节,并把其当作一个值取出来。
<addr>表示一个内存地址。
注意:严格区分n和u的关系,n表示单元个数,u表示每个单元的大小。
n/f/u三个参数可以一起使用。例如:
命令:x/3uh 0x54320 表示,从内存地址0x54320读取内容,h表示以双字节为一个单位,3表示输出三个单位,u表示按无符号十进制显示。
27》多线程调试中的一些技巧
(gdb)info threads
显示当前可调试的所有线程,每个线程会有一个GDB为其分配的ID,后面操作线程的时候会用到这个ID。
注意:前面有 * 的是当前调试的线程。
(gdb)thread ID
切换当前调试的线程为指定ID的线程。
(gdb)thread apply ID1 ID2 command
让一个或者多个线程执行GDB命令command。
(gdb)thread apply all command
让所有被调试线程执行GDB命令command。
(gdb)set scheduler-locking off|on|step
实际中发现,在使用step或者continue命令调试当前被调试线程的时候,其他线程也是同时执行的,怎么只让被调试程序执行呢?通过这个命令就可以实现。
off 不锁定任何线程,也就是所有线程都执行,这是默认值。
on 只有当前被调试程序会执行。
step 在单步的时候,除了next过一个函数的情况(熟悉情况的人可能知道,这其实是一个设置断点然后continue的行为)以外,只有当前线程会执行。
//显示线程堆栈信息
(gdb) bt
查看所有的调用栈
(gdb) f 3
调用框层次
(gdb) i locals
显示所有当前调用栈的所有变量
28》如何查看变量的内存地址
使用dispaly命令,如下:
dispaly &buff,查看变量buff的地址信息,或者display var,查看变量的值,或者通过p命令打印出变量的值
29》如何设置内存断点,使用命令watch,如下介绍使用方法。
int i; watch i, 查看变量i的具体数值
int *p; watch p,watch *p,
两种方式是不同的,不带星的查看的时watch *(&p),查看的是变量本身
带*号的是查看变量p所指内存的变量的值。常用于查看变量的具体的数据。
查看数组变量:
char buf[128];
watch buff;查看该128个数据的情况。
注:当使用watch命令查看一个局部变量是,当局部变量失效的时候,该断点也会自动失效,会显示如下信息:以下是本人试验所得信息:
Watchpoint 5 deleted because the program has left the block in
which its expression is valid.
30》gdb调试程序如何打印stl变量的内容。
在官方自带的gdb使用中使用调试程序,其在调试stl容器时,发现打印信息很不是十分的友好,导致有些信息是无法看到的。为了解决这种缺陷,听过了一种方式,可以很友好的帮助调试stl。
如上图所示,具体情况可根据自己实际使用情况而定。