您的位置:首页 > 其它

GDB 的使用详解

2014-02-26 16:27 218 查看
gdb使用

在linux下做东西,gcc,gdb是避不开的,刚开始使用比较麻烦,永久了感觉还是挺好用的。redhat里面自带一个gdb的图形前端kdbg,升级到高一点的版本还是挺好用的。还有一个xxgdb,没用过,不知道怎么样。

一、初始化

输入gdb进入gdb调试环境。或者直接输入gdb + progfile来加载文件。

注意该文件是使用gcc(或g++)编译得到的。为了使 gdb 正常工作, 必须

使你的程序在编译时包含调试信息,编译时必须使用-g参数来。

或者进入gdb环境后,通过命令file + progfile来加载需要调试的可

执行文件文件。

查看源代码:list [函数名][行数]

设置程序运行参数:set args

二、暂停程序

gdb可以使用几种方式来暂停程序:断点,观察点,捕捉点,信号,线

程停止。当程序被暂停后,可以使用continue、next、setp来继续执行程序。

continue 执行到下一暂停点或程序结束。

next 执行一行源代码但不进入函数内部。

setp 执行一行源代码而且进入函数内部。

1、设置断点:

a、break + [源代码行号][源代码函数名][内存地址]

b、break ... if condition ...可以是上述任一参数,condition

条件。例如在循环体中可以设置break ... if i = 100 来设置循环次数。

2、设置观察点:

a、watch + [变量][表达式] 当变量或表达式值改变时即停住程序。

b、rwatch + [变量][表达式] 当变量或表达式被读时,停住程序。

c、awatch + [变量][表达式] 当变量或表达式被读或被写时,停住程序。

3、设置捕捉点:

catch + event 当event发生时,停住程序。event可以是下面的内容:

1)、throw 一个C++抛出的异常。(throw为关键字)

2)、catch 一个C++捕捉到的异常。(catch为关键字)

3)、exec 调用系统调用exec时。(exec为关键字,目前此功能只在HP-UX下有用)

4)、fork 调用系统调用fork时。(fork为关键字,目前此功能只在HP-UX下有用)

5)、vfork 调用系统调用vfork时。(vfork为关键字,目前此功能只在HP-UX下有用)

6)、load 或 load 载入共享库(动态链接库)时。(load为关键字,

目前此功能只在HP-UX下有用)

7)、unload 或 unload 卸载共享库(动态链接库)时。(unload为关

键字,目前此功能只在HP-UX下有用)

4、捕捉信号:

handle + [argu] + signals

signals:是Linux/Unix定义的信号,SIGINT表示中断字符信号,也就是

Ctrl+C的信号,SIGBUS表示硬件故障的信号;SIGCHLD表示子进程状态改

变信号; SIGKILL表示终止程序运行的信号,等等。

argu:

nostop 当被调试的程序收到信号时,GDB不会停住程序的运行,但

会打出消息告诉你收到这种信号。

stop 当被调试的程序收到信号时,GDB会停住你的程序。

print 当被调试的程序收到信号时,GDB会显示出一条信息。

noprint 当被调试的程序收到信号时,GDB不会告诉你收到信号的信息。

pass or noignore 当被调试的程序收到信号时,GDB不处理信号。

这表示,GDB会把这个信号交给被调试程序会处理。

nopass or ignore 当被调试的程序收到信号时,GDB不会让被调

试程序来处理这个信号。

5、线程中断:

break [linespec] thread [threadno] [if ...]

linespec 断点设置所在的源代码的行号。如: test.c:12表示文件为

test.c中的第12行设置一个断点。

threadno 线程的ID。是GDB分配的,通过输入info threads来查看正在

运行中程序的线程信息。

if ... 设置中断条件。

三、查看信息

1、查看数据

print variable 查看变量

print
*array@len 查看数组(array是数组指针,len是需要数据长度)

可以通过添加参数来设置输出格式:

/x 按十六进制格式显示变量。

/d 按十进制格式显示变量。

/u 按十六进制格式显示无符号整型。

/o 按八进制格式显示变量。

/t 按二进制格式显示变量。

/a 按十六进制格式显示变量。

/c 按字符格式显示变量。

/f 按浮点数格式显示变量。

2、查看内存

examine /n f u + 内存地址(指针变量)

n 表示显示内存长度

f 表示输出格式(见上)

u 表示字节数制定(b 单字节;h 双字节;w 四字节;g 八字节;默认为四字节)

如:

x /10cw pFilePath (pFilePath为一个字符串指针,指针占4字节)

x 为examine命令的简写。

3、查看栈信息

backtrace [-n]

n 表示只打印栈顶上n层的栈信息。

-n 表示只打印栈底上n层的栈信息。

不加参数,表示打印所有栈信息。

附注:

基本gdb命令:

---------------------------------------------------------------------

命令 简写 功能

---------------------------------------------------------------------

file 装入想要调试的可执行文件.

kill k 终止正在调试的程序.

list l 列出产生执行文件的源代码的一部分.

next n 执行一行源代码但不进入函数内部.

step s 执行一行源代码而且进入函数内部.

continue c 继续执行程序,直至下一中断或者程序结束。

run r 执行当前被调试的程序.

quit q 终止 gdb.

watch 使你能监视一个变量的值而不管它何时被改变.

catch 设置捕捉点.

thread t 查看当前运行程序的线程信息.

break b 在代码里设置断点, 这将使程序执行到这里时被挂起.

make 使你能不退出 gdb 就可以重新产生可执行文件.

shell 使你能不离开 gdb 就执行 UNIX shell 命令.

print p 打印数据内容。

examine x 打印内存内容。

backtrace bt 查看函数调用栈的所有信息。

下面先说明GDB的基本指令:(使用时只要输入第一个字母就好了)

f(ile) :指定一个可执行文件进行调试,gdb将读取些文件的调试讯息,如f a.exe

l(ist):列程序出源文件

r(un):装载完要调试的可执行文件后,可以用run命令运行可执行文件

b(reak):设置断点(break point),如b 25,则在源程序的第25行设置一个断点,当程序执行到第25行时,就会产生中断;也可以使用b
funcname,funcname为函数的名称,当程序调用些函数时,则产生中断

c(ontinue):c命令可以另中断的程序继续执行,直到下一个中断点或程序结束

p(rint):输入某个变量的值,如程序定义了一个int aa的就是,p aa就会输出aa的当前值

n(ext):程序执行到断点时中断执行,可以用n指令进行单步执行

s(tep):程序执行到断点时中断执行,可以用s指令进行单步执行进某一函数

kill: 终止正在调试的程序

watch: 使你能监视一个变量的值而不管它何时改变

make: 使你能不退出gdb就可以重新产生可执行文件

shell:使你能不退出gdb就可以执行shell命令

q(uit):退出GDB

现在让我们来举一个简单的例子来说明GDB的使用,假设我们有以下的程序:

/****************************************************************************

gdb_sample.c

****************************************************************************/

#i nclude <stdio.h>

void PrintLn(const char* pMsg)

{

printf(“%s\n”, pMsg);

}

int main(int argc, char* argv[])

{

PrintLn(“Hello GDB”);

return 0;

}

调行以下命令编译程序gcc –g gdb_sample.c –o a.exe,生成a.exe的可执行文件。要用GDB调试程序,执行:
gdb a.exe
这样,我们就进入了gdb的调试环境。在gdb的命令行中输入list,gdb就会把上面的源程序打印出来,再次输入list,则进行翻页。接着,我们输入b
13
,表示在源程序的第13行PrintLn("Hello GDB")设置断点。设置断点后我们开始执行程序,在命令行中输入run,a.exe就开始执行。由于我们在第13行设置了断点,因此,程序会在PrintLn("Hello
GDB")处中断,此时,我们可以输入s,gdb会单步运行进PrintLn函数内,接着输入n,程序就会执行此语句:printf("%s\n",
pMsg),此时,我们再输入print pMsg,gdb就会输出pMsg的值:“Hello
GDB”。最后,我们输入c,程序继续执行,然后正常退出。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: