1.背景
程序的发布方式有两种,debug模式和release模式
Linux gcc/g++出来的二进制程序,默认是release模式
要使用gdb调试,必须在源代码生成二进制程序的时候, 加上 -g 选项
用gdb直接打开文件进行调试是不行的,因为这样gcc默认编译的版本是release版本,
而只有debug版本才能进行调试,gcc、g++默认形成的可执行程序是release的,不是debug
为什么会有debug版本和release版本呢?
这是因为debug版本里面含有需要对代码进行调试的文件信息,可供程序员进行查找错误用的
而release版本是程序员发行的版本,因为发行出来的文件不需要再加入调试文件等内容,要的是精简
所以release版本会去掉调试信息等文件来缩小体积
从上图我们就可以看见release版本的文件大小是比debug版本的大小更小的
通过readelf -S text-debug | grep -i debug的指令可以查看文件的调试信息
明显会看到release是不具备调试信息的。
如何用gcc将代码编译为debug版本的文件呢?
这里就需要用到gcc的一个选项:-g选项
在编译的时候添加-g选项,就是以debug版本进行发布的
2.使用
gdb 文件名开始进行调试,退出:ctrl+d或q
调试命令如下
list(l) 行号:显示binFile源代码,接着上次的位置往下列,每次列10行。
list(l) 函数名:列出某个函数的源代码。
r或run:运行程序。
n 或 next:单条执行。
s或step:进入函数调用
break(b) 行号:在某一行设置断点
break 函数名:在某个函数开头设置断点
info break :查看断点信息。
finish:执行到当前函数返回,然后挺下来等待命令
print(p):打印表达式的值,通过表达式可以修改变量的值或者调用函数
p 变量:打印变量值。
set var:修改变量的值
continue(或c):从当前位置开始连续而非单步执行程序
run(或r):从开始连续而非单步执行程序
delete breakpoints:删除所有断点
delete breakpoints n:删除序号为n的断点
disable breakpoints:禁用断点
enable breakpoints:启用断点
info(或i) breakpoints:参看当前设置了哪些断点
display 变量名:跟踪查看一个变量,每次停下来都显示它的值
undisplay:取消对先前设置的那些变量的跟踪
until X行号:跳至X行
breaktrace(或bt):查看各级函数调用及参数
info(i) locals:查看当前栈帧局部变量的值
quit(q):退出gdb
重点介绍
准备工作,这里用一段累加代码做演示
显示原代码指令:l(list)
输入l 1就是从第1行开始显示原代码,显示10行代码,每次显示10行代码
再次输入l (或者直接按回车,回车是执行最近一次的操作)会继续往下显示
运行程序的指令:r(run)从开始连续而非单步执行程序
对程序进行断点:b +行号,比如b16
查看断点:info b
这里可以看到有6个信息栏,分别是:Num:断点的序号,Type:类型,Disp:表现,Enb:是否开启
Address:存放信息的地址,What:断点存在位置
因为断点的信息有Enb这个开关,就是说断点是可以关闭的,叫做空心断点
有时我们像标记这个位置,但又不想让其断点生效,就可以用空心断点
用disable breakpoint +断点序号 进行关闭,关闭后的断点依然存在,但不会生效
开启:enable breakpoint +断点序号
删除断点:d+断电的序号
如上第一个断点的序号是1,断点再被删除后,它的序号是不会从零开始的,断点的序号是累加的!
gdb逐语句和逐过程
逐语句:一行一行代码的走,会进入函数
s(step):逐语句
逐过程:一行一行代码的走,但不会进入函数
n(next):逐过程
查看调试过程中变量的对应值:p
当然,这样每次查看都要用p进行未免叶太麻烦了,这里也提供了长显示功能:display
长显示功能:display
先用display对需要长显示的变量进行标记,然后进行逐过程或逐语句的时候就会紧接着在下面显示出来
取消长显示功能:undisplay +序号
取消长显示功能+的是序号,不是对应变量
长显示最开始的一行是其的序号
指令r 是重新开始调试
指令c是:从一个断点处直接运行值下一个断点处
指令finish:进入一个函数,只执行完该函数,就停下来
指令until +行号:在函数内,进行直到位置跳转,并执行完区间的代码