gdb调试
——用于发现程序中的逻辑错误。(语法错误由gcc编译器发现)
文章来源:C语言技术网(www.freecplus.net)+ B站视频课
作者:码农有道
链接:http://www.freecplus.net/b72113dda88a43b48728e0552fd8a74c.html
1、gdb安装
yum -y install gdb
2、编译参数设置
如果需要使用gdb调试程序,需要在编译的时候加上-g
gcc -g -o abc abc.c
3、gdb调试程序
gdb abc
4、基本调试命令
命令 | 命令缩写 | 命令说明 |
---|---|---|
set args | 设置主程序的参数。例如:./book119 /oracle/c/book1.c /tmp/book1.c 设置参数的方法是:gdb book119(gdb) set args /oracle/c/book1.c /tmp/book1.c | |
break | b | 设置断点,b 20 表示在第20行设置断点,可以设置多个断点。 info b 可以产看断点信息 |
run | r | 开始运行程序, 程序运行到断点的位置会停下来,如果没有遇到断点,程序一直运行下去。 |
next | n | 执行当前行语句,如果该语句为函数调用,不会进入函数内部执行。 |
step | s | 执行当前行语句,如果该语句为函数调用,则进入函数执行其中的第一条语句。注意了,如果函数是库函数或第三方提供的函数,用s也是进不去的,因为没有源代码,如果是您自定义的函数,只要有源码就可以进去。 |
p | 显示变量值,例如:p name表示显示变量name的值。 | |
continue | c | 继续程序的运行,直到遇到下一个断点。 |
set var name=value | 设置变量的值,假设程序有两个变量:int ii; char name[21];set var ii=10 把ii的值设置为10;set var name=“西施” 把name的值设置为"西施",注意,不是strcpy。 | |
quit | q | 退出gdb环境。 |
bt | 查看函数的调用栈 |
Note:可以用上下光标键选择执行过的gdb命令。
5、调试core文件
core文件介绍详见https://blog.csdn.net/qq_15821487/article/details/121132724
程序挂掉时,系统缺省不会生成core文件。
打开步骤:
1)ulimit -a 查看系统参数
2)ulimit -c unlimited //把core文件的大小设置为无限制
3)运行程序,生成core文件 //这里直接运行可执行程序即可
调试命令:
gdb abc core.XXX
此时会看到程序在哪一处出现问题。
6、调试正在运行中的程序
前提:使用两个终端窗口:一个窗口用于执行程序;另一个窗口用于调试程序。
1)使用命令【ps -ef|grep abc 】查看进程ID
2)使用命令【gdb abc -p 进程ID】调试程序
Note:在gdb调试台下进行调试时,正在运行的程序会被打断(暂停);可以看到程序在断断续续地执行。
7、调试多进程服务程序
在gdb中:
调试父进程:set follow-fork-mode parent(缺省)
调试子进程:set follow-fork-mode child
设置调试模式:set detach-on-fork [on|off]
缺省是on,表示调试当前进程的时候,其它进程继续运行;
如果是off,调试当前进程的时候,其它进程被gdb挂起;
查看调试的进程:info inferiors
切换当前调试的进程:inferior 进程ID
8、调试多线程服务程序
Note:编译时需要链接线程库 -lpthread
在终端中:
查看当前运行的进程 ps aux|grep abc
查看当前运行的轻量级进程 ps -aL|grep abc
查看主线程和子线程的关系:pstree -p 主线程ID
在gdb中:
查看线程:info threads
切换线程:thread 线程ID
只运行当前线程:set scheduler-locking on
运行全部的线程:set scheduler-locking off
指定某线程执行某gdb命令:thread apply 线程id cmd
全部的线程执行某gdb命令:thread apply all cmd
9、服务程序运行日志
设置断点或单步跟踪可能会严重干扰多进(线)程之间的竞争状态,导致我们看到的是一个假象。
一旦我们在某一个线程设置了断点,该线程在断点处停住了,只剩下另一个线程在跑,这时并发场景已经完全被破坏了,通过调试器看到的只是一个和谐的场景(理想状态)。
Q: 调试者的调试行为干扰了程序的运行,导致观察到的是一个干扰后的现象,该如何处理?
A: 输出log日志,可以避免断点和单步所导致的副作用。
小技巧:
(gdb) print *(address)
填充address指向的是一个16进制地址,输出是对应那个地址的代码的文件和行号。