gdb是linux下非常好用的一个调试工具,虽然它是命令行模式的调试工具,但是它的功能强大到你无法想象,这里简单介绍下gdb下常用的命令。

首先编译生成可执行文件(这里的test.c是一个简单的求前n项和的程序)。

gcc -g test.c -o test(-g选项告诉gcc在编译程序时加入调试信息)。
接下来可以这样。

gdb test

然后你就会看到出现好多信息在屏幕上,大致说的是gdb的一些版本信息说明之类的,但是它对你调试程序没用呀,所以,你可以加上-q选项,不输出它们。

gdb -q test

wang@king:~$ gdb -q test
Reading symbols from test...done.
(gdb) 

 

有没有觉得这个世界一下子清净了许多。

也可以先进入gdb模式,然后再加载文件。

wang@king:~$ gdb -q
(gdb) file test
Reading symbols from test...done.
(gdb) 

好了,现在开始调试了,但是我还想看看我的代码怎么办,gdb提供了一条命令,可以让你的程序显示出来。

(gdb)                                       //list默认一次显示10行
1    #include<stdio.h>
2    int func(int n)
3    {
4        int i;
5        int sum=0;
6        for(i=0;i<n;i++)
7            {
8                sum+=i;
9            }
10        return sum;
(gdb)                                       //直接输入回车重复上次命令,显示接下来的10行
11    }
12    int main()
13    {
14        int n;
15        printf("请输入n的值");
16        scanf("%d",&n);
17        printf("1+2+..+%d=%d",n,func(n));
18        return 0;
19    }
(gdb) 
            

list默认参数可以用show listsize来查看,如果感觉10行太多或者太少,还可以用set listsize <count>来更改。

list 还可以加上其他参数,比如:
list 5,10   显示第5行到第10行的代码;

list func   显示func函数周围的代码,显示范围和list参数有关;

list test.c:5,10  显示源文件test.c第5行到第10行的代码,一般用于调试含多个源文件的程序。

gdb 还支持字符串查找,search str,从当前行开始,向前查找含str的字符串;

reverse-search str,从当前行开始,向后查找含str的字符串。

现在你的屏幕应该被占满了吧?想清空屏幕,可是还在gdb里面呀,怎么办?其实,gdb也支持运行linux命令的,可以在gdb的提示符中,输入shell,然后在输入你需要的命令就可以了

(gdb) shell clear

这样也能达到清屏的效果。

看了程序的代码,感觉第6行代码可能有点问题,现在就需要我就需要设置一个断点,让程序停在第6行之前。

(gdb) break 6
Breakpoint 1 at 0x80484c8: file test.c, line 6.
(gdb) 

下面一行的 信息,1说明我设置的这个断点是第一个断点,断点所在内存地址为0x80484c8,它在文件test.c的第6行。

gdb还可以以条件表达式设置断点。

(gdb) break 7 if n==6
Breakpoint 2 at 0x80484d1: file test.c, line 7.
(gdb) 

这个断点的含义是,如果n的值为6,则程序运行到第7行停止。

当然,还可以直接在某个函数处设置断点;直接break 函数名就可以了,

然后我们想看下设置的断点信息,可以使用info breakpoints命令。

(gdb) info breakpoints
Num     Type           Disp Enb Address    What
1       breakpoint     keep y   0x080484c8 in func at test.c:6
2       breakpoint     keep y   0x080484d1 in func at test.c:7
    stop only if n==6
4       breakpoint     keep y   0x080484c1 in func at test.c:5
(gdb) 

Num表示断点的编号;Type表示断点的断点的类型,第二个断点类型还加上了条件;Disp表示中断点在执行一次之后是否失去作用,dis为是,keep为不是;Enb表示当前中断点是否有效,y为是,n为否;Address表示中断点所处的内存地址;What指出断点所处的位置。

如果不需要程序在该断点暂停时,有两种方法,一种是使该断点失效,一种是直接删除该断点。

(gdb) disable 1
(gdb) info breakpoints
Num     Type           Disp Enb Address    What
1       breakpoint     keep n   0x080484c8 in func at test.c:6
2       breakpoint     keep y   0x080484d1 in func at test.c:7
    stop only if n==6
3       breakpoint     keep y   0x080484c1 in func at test.c:5
(gdb) 

可以看到,第一个断点的Enb变为n了,表示该断点已经无效了,如果需要恢复,可以使用enable命令。这里需要注意的是,disable后面的参数为断点的编号。而不是行号。

直接删除该断点,可以使用clear命令和delete命令。

(gdb) clear 6
已删除的断点 1 
(gdb) 

clear命令后面的参数为设置断点的行号,clear后面参数还可以加设置断点的函数名。

delete命令后面的参数为断点的编号;可以一次删除多个断点,断点编号之间用空格隔开;如果delete后没有参数,默认删除所以断点,会给出提示选择是否操作。

(gdb) delete
删除所有断点吗? (y or n) 

断点设置好了,现在就可以调试了,

(gdb) run                                                //开始执行程序
Starting program: /home/wang/test 
请输入n的值10

Breakpoint 1, func (n=10) at test.c:6     //设置的第一个断点,程序在第6行暂停
6        for(i=0;i<n;i++)
(gdb) continue                                      //让程序继续运行,直到下个断点或者结束
Continuing.

Breakpoint 2, func (n=10) at test.c:8    //第二个断点设置的是i==6时停止
8                sum+=i;
(gdb) print i                                          //用print命令打印出i的值
$1 = 6
(gdb) print sum
$2 = 15
(gdb) next                                          //继续执行下一条语句,只执行一条。
6        for(i=0;i<n;i++)
(gdb) next    
8                sum+=i;
(gdb) print i
$3 = 7
(gdb) continue    
Continuing.
1+2+..+10=45[Inferior 1 (process 23636) exited normally]    
(gdb) quit                                             //退出gdb调试
                                            

 上面出现了很多命令,下面就来说说都是怎么用的。

run,开始运行程序;

continue,程序暂停时继续运行程序的命令;

print 变量名或表达式,打印该变量或者该表达式的值。whatis 变量名或者表达式,可以显示该变量或表达式的数据类型。

print  变量=值,这种形式还可以给对应的变量赋值;类似的还有set variable 变量=值。作用和用print赋值相同。

next,继续执行下一条语句;还有一条命令step,与之类似,不同的是,当下一条语句遇到函数调用的时候,next不会跟踪进入函数,而是继续执行下面的语句,而step命令则会跟踪进入函数内部。

(gdb) run
Starting program: /home/wang/test 

Breakpoint 1, main () at test.c:16
16        scanf("%d",&n);
(gdb) next
请输入n的值10  
17        printf("1+2+..+%d=%d",n,func(n));
(gdb) next                                             //next命令直接执行下一行,没有进入func函数  
18        return 0;
(gdb)

 

(gdb) run
Starting program: /home/wang/test 

Breakpoint 1, main () at test.c:16
16        scanf("%d",&n);
(gdb) n
请输入n的值10
17        printf("1+2+..+%d=%d",n,func(n));
(gdb) step                  //step命令跟踪进入了func函数  
func (n=10) at test.c:5
5        int sum=0;
(gdb) 

 还有nexti和stepi命令,这两个是单步执行一条机器指令,比如(i=0;i<n;i++)这条语句需要输入多个nexti才能执行完;两个的区别和上面相同。

quit,退出gdb调试,如果调试中想要退出,可以直接输入该命令,会出现提示选择是否退出。kill命令,结束当前程序的调试,(不会退出gdb)。

(gdb) quit
A debugging session is active.

    Inferior 1 [process 32229] will be killed.

Quit anyway? (y or n)