说在前面:
gdb所有的指令都可以使用简写来执行,比方说:continue 可以使用c 来表示
linux里面最终要的键,我觉得是TAB键。在gdb里面记得多按按,也许你就会发现不一样的天地。
如果你编写的程序需要能被gdb识别到,则需要添加-ggdb -g 【可供gdb识别到】 -O0 【不开启优化】
学习一个新命令的时候,多使用 --help 进行查看
使用gdb的场景:
- linux进行断点调试
- 查看段错误
- 查看当前进程的相关线程
- 配合ulimit使用 【这个也可以弄一篇文章】
- ... [你还想得起哪些场景呢?]
- android设备端的gdb【下篇讲】
如何使用gdb
有些设备需要自己编译gdb,再安装gdb。
请进入 官网 ,进行下载。
这里需要说明下他会关联termcap 如果没有要在下载这个包的源码进行编译
gdb相关的指令
指令 | 含义 | 例子 |
---|---|---|
gdb + 某一个应用 | gdb的方式启动某一个应用 | gdb test |
gdb + 某一个应用+ 某一个进程 | 挂载到某一个已经运行的引用 | gdb test 1234 |
gdb模式下:
开始
指令 | 含义 | 例子 |
---|---|---|
c | 继续执行,从暂停状态变成运行状态 | |
r | 重新开始 | |
q | 停止 | |
info threads | 显示出当前的线程列表 | |
r 参数列表 | 携带参数的方式重新运行 | r 1 2 3 |
断点
设置断点
指令 | 含义 | 例子 |
---|---|---|
b + 方法名 | 为方法设置断点 | b testA |
b + 文件 | 在某一个文件的某一行打上断点 | b a.c |
b + 文件 : 方法名 | 在某一个文件的某一方法打上断点, | b a.c:testA |
b + 设置 + if + expression | 条件成功的时候才设置断点 | b testA if a>0 |
取消断点
指令 | 含义 | 例子 |
---|---|---|
info breakpoints | 查看当前的断点信息 | |
delete [n] | n 表示断点NUM ,如果没有n 和clear的作用相似 | delete 1 |
clear | 清楚所有的断点 | |
disable | 不启用某个断点 | disable 1 |
enable | 启用某个断点 | enable 1 |
执行
指令 | 含义 | 例子 |
---|---|---|
n | 单步不进入,相当于 step over | |
s | 单步进入 ,相当于 step into | |
finish | 退出函数,相当于step out | |
c | 继续执行,相当于 Resume Program |
查看
指令 | 含义 | 例子 |
---|---|---|
bt | 在任何停止状态下,都可以打印出堆栈信息 | |
f + 栈号 | 进入到具体某一个堆栈里面的方法,此时就可以打印出相关的函数内部的值了,栈号通过bt查看 | f 2 |
p + 变量 | 打印出当前的变量的值 | p a |
回车 | 重复执行上一条指令 |
实例
查看哪些位置创建了线程
gdb test # gdb执行test
b pthread_create #将创建线程的函数打上日志
r #运行
...[卡住]
bt # 查看当前的堆栈信息,
info threads # 查看当前进程的线程
c # 继续运行
查看crash的位置在哪里
gdb test # gdb执行test
r
... [fatal error]
bt # 查看当前的堆栈信息,
f 2 # 跳转到具体的一个方法里面
p a # 查看在方法堆栈2里面a的值
问答
为啥,我用gdb的时候出现了 ??? 呢?
请确保你自己编译的库已经打开了-g -ggdb -O0,
有些第三方库,是不开启这个选项的,所以如果gdb打印的堆栈到了那些库里面,就会出现???,比方说qt相关的库