Linux调试器-gdb介绍

Source

image-20221217215247988

gdb的基础使用

gdb是什么

GDB是 GNU开源组织发布的一个强大的Unix/Linux下的,基于命令行的程序调试工具。

以下gdb的使用是基于像知道在vs上调试思路和方法基础上讲解的(本人是用的vs stdio2019)

gdb的使用

gdb的下载

指令:sudo yum -y install gdb

输入后需要输入当前用户的密码

image-20221217151040522

在gdb模式下输入

q(或者ctrl+d) 退出gdb调试模式
l(list) 显示binFile源代码,一次显示10行按enter接着显示
l 0 从第0行开始显示,按enter接着显示
b(breakpoint) 行号 在目标行打断点-形成序号
info b 查看断点-显示序号
d 断点序号 删除序号的断点-删除序号
r(run) 从开始连续而非单步执行程序-简称运行调试->等同于vs 里面的F5(开始调试)
n(next) 逐过程
s(step) 逐语句
c(continue) 从当前位置开始连续而非单步执行程序即是到下一个断点停下
bt(breaktrace) 查看各级函数调用及参数
finish 执行到当前函数返回,然后停下来等待命令
p 变量 打印变量值
disable 变量名 常显示变量
undisable 序号 取消常显示序号对应的变量
until 行号 跳到对应行
set var 变量名 修改变量的值
info locals 查看当前栈帧局部变量的值
enable breakpoints 序号 启用序号对应的断点
disable breakpoints 序号 禁用序号对应的断点(断点只有标记的用处)

然后我们创建一个test.c和一个Makefile,前者写一个从0加到100的函数并打印,后者编译test.c生成可执行程序test

image-20221217152709202

并且能够执行(timestamp-时间戳)

image-20221217152932045

然后我们调试test 输入gdb 文件名称 进入gdb调试模式

image-20221217153522830

并且我们还发现下面写着没有找到bug标记,并且调试不了!为啥?(输入q退出gdb模式)

在Linux中,用gcc或g++生产程序是默认是release版本,并且是动态链接库,而gdb无法对release版本进行调试,gdb对debug版本才能进行调试。那么我们需要把release版本换成debug版本。

我们只需在编译时在程序后加-g即可。这里我为了更可观的看到调试在可执行程序后面加了_g后缀。

我们进入Makefile文件修改

image-20221217155559551

我们可以看到debug版本比release版本大小要大

release版本面向用户,那么只需要最优化的版本。而debug版本面向程序员,需要有调试等等操作的文件在里面。

image-20221217155727694

我们通过readelf -S 文件名 查看可执行程序条带化对应的可执行程序细节信息

我们通过 readelf -S test_g(debug版本) 和readelf -S test(release版本)查看二者可发现debug版本下就多了图中这些调试信息

image-20221217161210892

这样可以更直观的看到二者区别

image-20221217161602842

l 显示代码

image-20221217162304532

我们接着按enter就会接着显示直到显示完全

image-20221217162551622

b 行号 :打断点

我们在第11行,13行,15行打断点

info b :查看断点

d 断点序号 :删除断点

我们删除 第1个断点 第2个断点

image-20221217163515275

r :运行调试

我们运行调试到断点1-代码的第19行停下

image-20221217184752797

n(next) :逐过程

现在我们不进入Addsum函数并想把这个函数跑完到下一个行,那么我们可以输入n(next)

image-20221217185522625

s(step):逐语句

现在我们进入Print函数那么可以输入s

image-20221217185756320

现在我们输入info b查看断点,可以看到断点1已经被命中一次

image-20221217190325100

现在我给test.c加一些代码

image-20221217192830543

c(continue):从这个断点到下一个断点停下

现在我在第19、23、26、27行打断点,run到19行后,我不想进入Add函数想跑到下一个断点,可以输入c到23行

image-20221217193028020

bt:查看函数栈帧

相同的我先run到第19行,进入Addsum函数,输入bt可以看到函数栈帧

image-20221217204812552

finish 执行到当前函数返回,然后听下来等待命令

如上图我进入Add函数后,输入finish把函数跑完并停下来,再输入n才到第20行

p 临时变量名

现在我run到19行进到Addsum函数体内,输入p i查看临时变量i ,查看 i地址输入p &i

image-20221217210822101

display 临时变量名:常显示临时变量 undisplay 序号 :取消常显示 序号对应的临时变量

常显示 变量i,变量i的地址,变量sum,取消常显示 序号1(i)

image-20221217211609798

until 行号 :跳到某行

image-20221217212004973

set var 临时变量名:修改临时变量

同样的我们r到19行进入函数Addsum函数,原本变量i=0,set var i=99后,临时变量变为99了,同样函数最后的结果也会改变!

image-20221217212640370

info locals :查看当前栈帧局部变量的值

进入Addsum函数体内,info locals查看到查看当前栈帧局部变量的值为sum=0

image-20221217213001184

enable breakpoints:启用断点 disable breakpoints:禁用断点

我们再一次调试它,取了三个断点,查看后都是启用的,输入disable breakpoints 1->禁用断点1,再enable breakpoints 1启用断点1

**image-20221217213712958
ok对于Linux调试器gdb的介绍就到这了,介绍了基础实用的指令,要多多练习才能熟记于心噢~