Linux中 包含有一个很有用的调试工具--gdb(GNU Debuger),它可以用来调试C和C++程序,功能不亚于Windows下的许多图形界面的调试工具。

和所有常用的调试工具一 样,gdb提供了以下功能:
 # 监视程序中变量的值
 # 在程序中设置断点
 # 程序的单步执行

在使用gdb前,必须先载入可执行文件,因为要进行调试,文件中就必须包含调 试信息,所以在用gcc或cc编译时就需要用-g参数来打开程序的调试选项。


调试开始时,必须先载入要 进行调试的程序,可以用以下两种方式:
 * 在启动gdb后执 行以下命令:
    file 可执行文件路径
 * 在gdb启动时就载入程序:
   gdb 可执行文件路径

GDB使用介绍:

查看源文件
1. List
执行程序
要想运行准备调试的程序,可使用run命令,在它后面可以跟随发给该程序的任何参数,包括标准输入和标准输出说明符(<和>)和外壳通配符(*、?、[、])在内。
如果你使用不带参数的run命令,gdb就再次使用你给予前一条run命令的参数,这是很有用的。
利用set args 命令就可以修改发送给程序的参数,而使用show args 命令就可以查看其缺省参数的列表。
(gdb)set args –b –x
(gdb) show args
backtrace命令为堆栈提供向后跟踪功能。
Backtrace 命令产生一张列表,包含着从最近的过程开始的所以有效过程和调用这些过程的参数。

显示数据
int ch = 'a';
要查看c的值 (gdb) p ch
用指定的类型查看:
(gdb) p/c ch

print 是gdb的一个功能很强的命令,利用它可以显示被调试的语言中任何有效的表达式。表达式除了包含你程序中的变量外,还可以包含以下内容:
l 对程序中函数的调用
(gdb) print find_entry(1,0)
l 数据结构和其他复杂对象
(gdb) print *table_start
$8={e=reference=’00’,location=0x0,next=0x0}
l 值的历史成分
(gdb)print $1 ($1为历史记录变量,在以后可以直接引用 $1 的值)
l 人为数组
人为数组提供了一种去显示存储器块(数组节或动态分配的存储区)内容的方法。早期的调试程序没有很好的方法将任意的指针换成一个数组。就像对待参数一样,让我们查看内存中在变量h后面的10个整数,一个动态数组的语法如下所示:
base@length
因此,要想显示在h后面的10个元素,可以使用h@10:
(gdb)print h@10
$13=(-1,345,23,-234,0,0,0,98,345,10)

断点(breakpoint)
break命令(可以简写为b)可以用来在调试的程序中设置断点,该命令有如下四种形式:
l break line-number 使程序恰好在执行给定行之前停止。
l break function-name 使程序恰好在进入指定的函数之前停止。
l break line-or-function if condition 如果condition(条件)是真,程序到达指定行或函数时停止。
l break routine-name 在指定例程的入口处设置断点

如果该程序是由很多原文件构成的,你可以在各个原文件中设置断点,而不是在当前的原文件中设置断点,其方法如下:
(gdb) break filename:line-number
(gdb) break filename:function-name

要想设置一个条件断点,可以利用break if命令,如下所示:
(gdb) break line-or-function if expr
例:
(gdb) break 46 if testsize==100

从断点继续运行:countinue 命令

断点的管理
1. 显示当前gdb的断点信息:
(gdb) info break
他会以如下的形式显示所有的断点信息:
Num Type Disp Enb Address What
1 breakpoint keep y 0x000028bc in init_random at qsort2.c:155
2 breakpoint keep y 0x0000291c in init_organ at qsort2.c:168
(gdb)
2.删除指定的某个断点:
(gdb) delete breakpoint 1
该命令将会删除编号为1的断点,如果不带编号参数,将删除所有的断点
(gdb) delete breakpoint
3.禁止使用某个断点
(gdb) disable breakpoint 1
该命令将禁止断点 1,同时断点信息的 (Enb)域将变为 n
4.允许使用某个断点
(gdb) enable breakpoint 1
该命令将允许断点 1,同时断点信息的 (Enb)域将变为 y
5.清除原文件中某一代码行上的所有断点
(gdb)clean number
注:number 为原文件的某个代码行的行号
六.变量的检查和赋值
l whatis:识别数组或变量的类型
l ptype:比whatis的功能更强,他可以提供一个结构的定义
l set variable:将值赋予变量
l print 除了显示一个变量的值外,还可以用来赋值

单步执行
l next
不进入的单步执行
l step
进入的单步执行
如果已经进入了某函数,而想退出该函数返回到它的调用函数中,可使用命令finish

函数的调用
l call name 调用和执行一个函数
(gdb) call gen_and_sork( 1234,1,0 )
(gdb) call printf(“abcd”)
$1=4
l finish 结束执行当前函数,显示其返回值(如果有的话)

原文件的搜索
search text:该命令可显示在当前文件中包含text串的下一行。
Reverse-search text:该命令可以显示包含text 的前一行。

shell 命令
shell 命令可启动UNIX外壳,CTRL-D退出外壳,返回到 gdb.

命令的历史
为了允许使用历史命令,可使用 set history expansion on 命令
(gdb) set history expansion on


基本的常用指令:

   

file FILE 装载指定的可执行文件进行调试。

kill 终止正在调试的程序.

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

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

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

run 执行当前被调试的程序

quit 终止 gdb

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

make 在不退出 gdb 的情况下运行 make 工具。

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


break NUM 在指定的行上设置断点, 这将使程序执行到这里时被挂起.

bt 显示所有的调用栈帧。该命令可用来显示函数的调用顺序。

clear 删除设置在特定源文件、特定行上的断点。其用法为clear FILENAME:NUM

continue 继续执行正在调试的程序。该命令用在程序由于处理信号或断点而 导致停止运行时。

display EXPR 每次程序停止后显示表达式的值。表达式由程序定义的变量组成。

print EXPR 显示表达式 EXPR 的值。


info break 显示当前断点清单,包括到达断点处的次数等。

info files 显示被调试文件的详细信息。

info func 显示所有的函数名称。

info local 显示当函数中的局部变量信息。

info prog 显示被调试程序的执行状态。

info var 显示所有的全局和静态变量名称。

help NAME 显示指定命令的帮助信息。



更多命令,以及部分命令的详细介绍:

 * list:显示程序中的代码,常用使用格式有:

    list

      输出从上次调用list命令开始往后的10行程序代码。

    list -

      输出从上次调用list命令开始往前的10行程序代码。

    list n

      输出第n行附近的10行程序代码。

    list function

      输出函数function前后的10行程序代码。

 * forward/search:从当前行向后查找匹配某个字符串的程序行。使用格式:

    forward/search 字符串

  查找到的行号将保存在$_变量中,可以用print $_命令来查看。

 * reverse-search:和forward/search相反,向前查找字符串。使用格式同上。

 * break:在程序中设置断点,当程序运行到指定行上时,会暂停执行。使用 格式:

     break 要 设置断点的行号

 * tbreak:设置临时断点,在设置之后只起作用一次。使用格式:

    tbreak 要设置临时断点的行号

 * clear:和break相反,clear用于清除断点。使用格式:

    clear 要清除的断点所在的行号

 * run:启动程序,在run后面带上参数可以传递给正在调试的程序。

 * awatch:用来增加一个观察点(add watch),使用格式:

    awatch 变量或表达式

  当表达式的值发生改变或表达式的值被读取时,程序就会停止运 行。

 * watch:与awatch类似用来设置观察点,但程序只有当表达式的值发生改变时才会停止 运行。使用格 式:

    watch 变量或表达式

  需要注意的是,awatch和watch都必须在程序运行的过程中设置观察点,即可运行run之后才能设置。

 * commands:设置在遇到断点后执行特定的指令。使用格式有:

    commands

      设置遇到最后一个遇到的断点时要执行的命令

    commands n

      设置遇到断点号n时要执行的命令

  注意,commands后面跟的是断点号,而不是断点所在的行号。

  在输入命令后,就可以输入遇到断点后要执行的命令,每行一条 命令,在输入最后一条命令后输入end就 可以结束输入。

 * delete:清除断点或自动显示的表达式。使用格式:

    delete 断点号

 * disable:让指定断点失效。使用格式:

    disable 断点号列表

  断点号之间用空格间隔开。

 * enable:和disable相反,恢复失效的断点。使用格式:

    enable 断点编号列表

 * ignore:忽略断点。使用格式:

    ignore 断点号 忽略次数

 * condition:设置断点在一定条件下才能生效。使用格式:

    condition 断点号 条件表达式

 * cont/continue:使程序在暂停在断点之后继续运行。使用格式:

    cont

      跳过当前断点继续运行。

    cont n

      跳过n次断点,继续运行。

  当n为1时,cont 1即为cont。

 * jump:让程序跳到指定行开始调试。使用格式:

    jump 行号

 * next:继续执行语句,但是跳过子程序的调用。使用格式:

    next

      执行一条语句

    next n

      执行n条语句

 * nexti:单步执行语句,但和next不同的是,它会跟踪到子程序的内部,但不打印出子程序内部的语 句。使用格式同上。

 * step:与next类似,但是它会跟踪到子程序的内部,而且会显示子程序内部的执行 情况。使用格式同上。

 * stepi:与step类似,但是比step更详细,是nexti和step的结合。使用格式同上。

 * whatis:显示某个变量或表达式的数据类型。使用格式:

    whatis 变量或表达式

 * ptype:和whatis类似,用于显示数据类型,但是它还可以显示typedef定义的类型等。使用格式:

    ptype 变量或表达式

 * set:设置程序中变量的值。使用格式:

    set 变量=表达式

    set 变量:=表达式

 * display:增加要显示值的表达式。使用格式:

    display 表达式

 * info display:显示当前所有的要显示值的表达式。

 * delete display/undisplay:删除要显示值的表达式。使用格式:

    delete display/undisplay 表达式编号

 * disable display:暂 时不显示一个要表达式的值。使用格式:

    disable display 表达式编号

 * enable display:与disable display相反,使用表达式恢复显 示。使用格式:

     enable display 表达式编号

 * print:打印变量或表达式的值。使用格式:

    print 变量或表达式

  表达式中有两个符号有特殊含义:$和$$。

  $表示给定序号的前一个序号,$$表示给定序号的前两个序号。

  如果$和$$后面不带数字,则给定序号为当前序号。

 * backtrace:打印指定个数的栈帧(stack frame)。使用格式:

    backtrace 栈帧个数

 * frame:打印栈帧。使用格式:

    frame 栈帧号

 * info frame:显示当前栈帧的详细信息。

 * select-frame:选择栈帧,选择后可以用info frame来显示栈帧信息。使用格式:

    select-frame 栈帧号

 * kill:结束当前程序的调试。

 * quit:退出gdb。