GDB调试从基础到精通实例

来源:互联网 发布:巫师3解压数据错误 编辑:程序博客网 时间:2024/05/29 12:35
GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具。或许,各位比较喜欢那种图形界面方式的,像VC、BCB等IDE的调试,但如果你是在UNIX平台下做软件,你会发现GDB这个调试工具有比VC、BCB的图形化调试器更强大的功能。所谓“寸有所长,尺有所短”就是这个道理。
 
一般来说,GDB主要帮忙你完成下面四个方面的功能:
 
    1、启动你的程序,可以按照你的自定义的要求随心所欲的运行程序。
    2、可让被调试的程序在你所指定的调置的断点处停住。(断点可以是条件表达式)
    3、当程序被停住时,可以检查此时你的程序中所发生的事。
    4、动态的改变你程序的执行环境。
 
从上面看来,GDB和一般的调试工具没有什么两样,基本上也是完成这些功能,不过在细节上,你会发现GDB这个调试工具的强大,大家可能比较习惯了图形化的调试工具,但有时候,命令行的调试工具却有着图形化工具所不能完成的功能。让我们一一看来。
 
使用GDB<br>————<br><br>一般来说GDB主要调试的是C/C++的程序。要调试C/C++的程序,首先在编译时,我们必须要把调试信息加到可执行文件中。使用编译器(cc/gcc/g++)的 -g 参数可以做到这一点。如:<br><br>    > cc -g hello.c -o hello<br>    > g++ -g hello.cpp -o hello<br><br>如果没有-g,你将看不见程序的函数名、变量名,所代替的全是运行时的内存地址。当你用-g把调试信息加入之后,并成功编译目标代码以后,让我们来看看如何用gdb来调试他。<br><br>启动GDB的方法有以下几种:<br><br>    1、gdb <program> <br>       program也就是你的执行文件,一般在当然目录下。<br><br>    2、gdb <program> core<br>       用gdb同时调试一个运行程序和core文件,core是程序非法执行后core dump后产生的文件。<br><br>    3、gdb <program> <PID><br>       如果你的程序是一个服务程序,那么你可以指定这个服务程序运行时的进程ID。gdb会自动attach上去,并调试他。program应该在PATH环境变量中搜索得到。
<br><br>一个调试示例
——————
 
源程序:tst.c
 
     1 #include <stdio.h>
     2
     3int func(intn)
     4 {
     5        int sum=0,i;
     6        for(i=0; i<n; i++)
     7         {
     8                 sum+=i;
     9         }
    10        return sum;
    11 }
    12
    13
    14 main()
    15 {
    16        int i;
    17        long result = 0;
    18        for(i=1; i<=100; i++)
    19         {
    20                 result += i;
    21         }
    22
    23       printf("result[1-100] = %d /n", result );
    24       printf("result[1-250] = %d /n", func(250) );
    25 }
 
编译生成执行文件:(Linux下)
    hchen/test> cc -g tst.c -o tst
 
使用GDB调试:
 
hchen/test> gdb tst  <---------- 启动GDB
GNU gdb 5.1.1
Copyright 2002 Free Software Foundation, Inc.
GDB is freesoftware, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying"to see the conditions.
There is absolutely no warrantyfor GDB.  Type"show warranty" for details.
This GDB was configured as"i386-suse-linux"...
(gdb) l     <-------------------- l命令相当于list,从第一行开始例出原码。
1        #include <stdio.h>
2
3        intfunc(int n)
4        {
5                int sum=0,i;
6                for(i=0; i<n; i++)<---------------  ptype  i  打印变量的类型  p 数组名,打印数组元素  p &数组名 打印地址   p i=7 改变运行程序变量的值<br>7                {
8                        sum+=i;<-----------------------until 退出循环   bif i=5打断点  watch i i值发送改变是打断点<br>9                }
10               return sum;<----------------------------------finish 退出真个函数<br>(gdb)       <-------------------- 直接回车表示,重复上一次命令
11       }
12
13
14       main()
15       {
16               int i;
17               long result = 0;
18               for(i=1; i<=100; i++)
19               {
20                       result += i;   
(gdb) break16    <-------------------- 设置断点,在源程序第16行处。
Breakpoint 1 at 0x8048496: file tst.c, line 16.
(gdb) breakfunc  <-------------------- 设置断点,在函数func()入口处。
Breakpoint 2 at 0x8048456: file tst.c, line 5.
(gdb) info break  <-------------------- 查看断点信息。
Num Type           Disp Enb Address    What
1   breakpoint     keep y   0x08048496 in main at tst.c:16
2   breakpoint     keep y   0x08048456 in func at tst.c:5
(gdb) r           <--------------------- 运行程序,run命令简写
Starting program: /home/hchen/test/tst
 
Breakpoint 1, main () at tst.c:17    <---------- 在断点处停住。
17               long result = 0;
(gdb) n          <--------------------- 单条语句执行,next命令简写。
18               for(i=1; i<=100; i++)
(gdb) n
20                       result += i;
(gdb) n
18               for(i=1; i<=100; i++)
(gdb) n
20                       result += i;
(gdb) c          <--------------------- 继续运行程序,continue命令简写。
Continuing.
result[1-100] = 5050       <----------程序输出。
 
Breakpoint 2, func (n=250) at tst.c:5
5               int sum=0,i;
(gdb) n
6               for(i=1; i<=n; i++)
(gdb) p i        <--------------------- 打印变量i的值,print命令简写。
$1 = 134513808
(gdb) n
8                        sum+=i;
(gdb) n
6               for(i=1; i<=n; i++)
(gdb) p sum
$2 = 1
(gdb) n
8                        sum+=i;
(gdb) p i
$3 = 2
(gdb) n
6               for(i=1; i<=n; i++)
(gdb) p sum
$4 = 3
(gdb) bt        <--------------------- 查看函数堆栈。
#0  func (n=250) at tst.c:5
#1  0x080484e4 in main () at tst.c:24
#2  0x400409ed in __libc_start_main () from /lib/libc.so.6
(gdb) finish    <--------------------- 退出函数。
Run till exit from #0  func (n=250) at tst.c:5
0x080484e4 in main () at tst.c:24
24             printf("result[1-250] = %d /n", func(250) );
Value returned is $6 = 31375
(gdb) c     <--------------------- 继续运行。
Continuing.
result[1-250] = 31375    <----------程序输出。
 
Program exited with code 027. <--------程序退出,调试结束。
(gdb) q     <--------------------- 退出gdb。
hchen/test>
<br>掌握以上知识点,就具备了GDB调试程序的基本能力。
  
GDB启动时,可以加上一些GDB的启动开关,详细的开关可以用gdb -help查看。我在下面只例举一些比较常用的参数:
 
    -symbols <file>
    -s <file>
    从指定文件中读取符号表。
 
    -se file
    从指定文件中读取符号表信息,并把他用在可执行文件中。
 
    -core <file>
    -c <file>
    调试时core dump的core文件。
 
    -directory <directory>
    -d <directory>
    加入一个源文件的搜索路径。默认搜索路径是环境变量中PATH所定义的路径。
<br><br><br></span>
0 0