gdb学习

概述

gdb是GNU debugger的缩写,是编程调试工具。

功能

1、启动程序,可以按照用户自定义的要求随心所欲的运行程序。

2、可让被调试的程序在用户所指定的断点处停住 (断点可以是条件表达式)。

3、当程序停住时,可以检查此时程序中所发生的事。

4、动态改变程序的执行环境。

示例:

gcc -Wall -g simple.c -o simple

#一定要记得加上-g选项,以便生成调试信息

启动gdb:gdb [-q] [executable-file] [core-file]

gdb simple

#启动gdb

list(l)

#查看程序的十行代码,连续键入list命令,可查看该文件所有代码

break(b) n

#在第n行设置断点

info break(i b)

#查看断点设置信息

break function-name (b function-name)

#在函数的入口处设置断点

run(r)

#启动程序,在断点处暂停

step(s)

#单步跟踪,一步步走

print i(p i)

#打印i当前值

until

#跳出循环

continue(c)

#继续运行,直到下一个断点,如果一直运行到程序结束,则会打印程序结束信息

enter键(回车)

#表示使用上一个命令

finish

#退出当前函数,打印函数的返回值

delete n

#删除断点n

quit(q)

#退出整个gdb调试程序

运行程序

1、run(r) #运行程序,直到第一个断点或程序结束

2、run arg1 arg2 … #运行程序并加上参数,如:r a b c d ...

查看源码

list(l) - 查看最近10行源码

list fun-name - 查看fun函数源码

list file:fun-name – 查看file文件中的fun函数源码

如:l search.c:seq_search #如果只有一个文件包含有seq_search函数,则不必加”search.c:”

list [m,n]

list 2,10 – 查看从第2行到第10行源代码

list ,15 – 查看最多到第15行的源代码

设置断点与观察点

break [line-num]

break [fun-name]

break [file-name:line-num]   #b search.c:4

break [file-name:fun-name]

break if <condition> - 条件成立时程序停住

E.g. break if i = 10

info break(i b) – 查看断点

watch expr – 一旦expr值发生改变,程序就停住。【设置观察点】

E.g. watch i   #一旦i值发生改变,则程序暂停

delete n(d n) – 删除断点n # d 1 d 1 2 3

单步调试

continue(c) 

- 运行至下一个断点

step(s) 

- 单步跟踪,进入函数内部,类似于Vs中的step in[但是不会进入没有提供源代码的函数]

next(n) 

- 单步跟踪,不进入函数内部,类似于VC中的step out

finish 

- 运行程序,直到当前函数完成返回。并打印函数返回时的堆栈地址和返回值及参数值等信息。但是不会跳出main函数。

until 

– 运行程序直到退出循环体。如果一次没有跳出,则执行下一次

return [value]

- 停止当前函数,提前让函数返回[动态调整程序的执行],将value返回调用者,相当于step return

set variable var-name = value

- 改变一个变量的值

gdb常用命令总结

命令

作用

run(r)

运行,启动程序

list(l)

列出10行源程序

break(b)

设置断点

info break(i b)

打印断点详细信息

continue(c)

继续运行程序,直到下一个断点

watch

设置观察点

step(s)

单步跟踪,类似于VC中的step in

next(n)

单步跟踪,类似于VC中的step out

finish

运行程序,直到当前函数返回,返回函数

until(u)

跳出循环(有时需要两次)

print(p)

查看运行时变量及表达式

help

打印帮助信息

附-

#Makefile

CC = gcc 

CFLAGS = -Wall -g

BIN = sample main 

.PHONY: all clean

all: $(BIN)

%.o: %.c 

    $(CC) $(CFLAGS) -c   $<   -o   $@      #c文件生成o文件

main: main.o search.o

    $(CC)    $(CFLAGS)     $^  -o   $@

clean:

    -rm -rf    $(BIN)    *.o

查看运行时数据

print - 查看变量值

ptype – 查看变量类型

#ptype i

#ptype “aaa” 打印字符串”aaa”的类型

#ptype array 打印数组array的类型

#ptype main 打印main函数的类型

print array – 查看数组

#p arr1

#p arr1[2] 查看数组arr2第三个元素值

#p &arr1[2] 打印数组元素的地址

#p &arr1 打印数组首地址,与p &arr1[0]作用类似

print *array@len – 查看动态内存

#p *arr2@10 

#p *arr2@15  #将数组越界之后的内存也打印出来了

#p *arr@10 #静态的数组也可以改方式打印

#p arr[2]@2 #从arr[2]开始的两个元素

#p arr@2 #打印两个数组

print x=5 – 动态改变运行时数据

#print i = 1000

程序错误

编译错误:编写程序的时候没有符合语言规范导致编译错误【语法错误】。

运行时错误:编译器检查不出这种错误,但在运行的时候可能会导致程序崩溃【如:内存地址非法访问】。

逻辑错误:编译和运行都很顺利,但是执行结果不对

gdb调试逻辑错误

 
//示例-实现逆序输出字符串hello,但是却没有任何输出
#include <stdio.h>
int main(void)
{
        int i;  
        char str[6] = "hello";
        char reverse_str[6] = "";
 
        printf("%s\n", str);
        for (i=0; i<5; i++)
                reverse_str[5-i] = str[i]; 	
//Error:应该是reverse_str[4-i]...
 
        printf("%s\n", reverse_str);
        return 0;
}

gdb调试段错误

段错误是由于访问非法地址而产生的错误。

1、访问系统数据区,尤其是往系统保护的内存地址写数据。最常见就是给一个0地址指针赋值 

2、内存越界(数组越界,变量类型不一致等) 访问到不属于你的内存区域

where命令 or bt命令:栈回溯,显示导致段错误的执行函数树

 wh命令查看程序代码窗口

core文件调试

1、core文件

在程序崩溃时,一般会生成一个文件叫core文件。core文件记录的是程序崩溃时的内存映像,并加入调试信息。core文件生成的过程叫做core dump

2、设置生成core文件

ulimit -c   #查看core-dump状态

ulimit -c 数字 (如:ulimit -c 1024)

ulimit -c unlimited

将ulimit -c unlimited写入~/.bashrc,则默认会生成core文件

3、gdb利用core文件调试

gdb 程序名 [core-file] #如:gdb bugging core

猜你喜欢

转载自blog.csdn.net/alatebloomer/article/details/81185816
GDB
今日推荐