linux环境的gdb使用法。

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。

网上提供了一种方法是:http://www.yolinux.com/TUTORIALS/src/dbinit_stl_views-1.03.txt文件的内容拷贝出来,存放到一个.txt文件,之后更名为.gdbinit,再讲其移动到主目录 ~/.gdbinit下。之后就可以直接使用来调试stl。例如:可通过help来查看相关的命令使用方法;

如上图所示,具体情况可根据自己实际使用情况而定。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值