【Linux系统编程】第十五弹---调试器gdb使用

2024-05-09 13:36

本文主要是介绍【Linux系统编程】第十五弹---调试器gdb使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

个人主页: 熬夜学编程的小林

💗系列专栏: 【C语言详解】 【数据结构详解】【C++详解】【Linux系统编程】

目录

1、背景

2、安装gdb

 3、gdb的使用

总结


1、背景

前面我们学习了文件编辑器,项目自动化构建工具,以及gcc编译器,但是在我们写代码的过程中不仅仅只有写代码,还有改bug的阶段,在VS在我们通过debug模式对代码进行调试,那么我们Linux中如何调试代码呢???

这就用到了我们这弹要学习的gdb调试器了。

必要知识:

程序的发布方式有两种,debug模式(能调试)和release模式(不能调试)。
Linux gcc/g++出来的二进制程序,默认是release模式
要使用gdb调试,必须在源代码生成二进制程序的时候, 加上 -g 选项。

★ debug版本可调试,那是因为编译器形成可执行程序的时候,会给可执行程序添加添加调试信息;而release版本不能调试。


2、安装gdb

根据我们前面讲解的几个工具,基本上都是需要先进行安装的,gdb也不例外。

如果没有安装,则会出现下面的提示。

[jkl@VMCentos7 dir.gdb]$ gdb
-bash: /usr/bin/gdb: No such file or directory  # 没有该文件或目录

 gdb的下载跟我们前面下载的工具也基本类似,使用yum工具。此处博主用的普通用户,需要加sudo。

sudo yum install -y gdb    # 输入该命令  

出现下面的样子就安装成功了。

 3、gdb的使用

在使用gdb之前得先编写一个C语言代码,并使用-g(debug)编译,才能进行调试。

1. 创建一个文件并编写相应的代码。

[jkl@VMCentos7 dir.gdb]$ touch test.c # 创建文件
[jkl@VMCentos7 dir.gdb]$ vim test.c   # 编写代码
[jkl@VMCentos7 dir.gdb]$ cat test.c   # 查看代码
#include<stdio.h>
int main()
{int i=0;for(i=0;i<10;i++){printf("%d\n",i);}return 0;
}

2. 以debug模式进行编译。

[jkl@VMCentos7 dir.gdb]$ gcc -o test-debug test.c -g  # debug模式需加-g
[jkl@VMCentos7 dir.gdb]$ ls
test.c  test-debug   # test-debug为可执行程序

 3. 如何确认为debug模式?

a. 对比release版本和debug版本可执行程序的大小。

因为debug版本会添加调试信息,必然文件大小比release版本大。

[jkl@VMCentos7 dir.gdb]$ gcc -o test test.c  # release版本生成可执行程序
[jkl@VMCentos7 dir.gdb]$ ll
total 60
-rwxrwxr-x 1 jkl jkl 25792 Apr  4 14:15 test
-rw-rw-r-- 1 jkl jkl   121 Apr  4 14:06 test.c
-rwxrwxr-x 1 jkl jkl 27008 Apr  4 14:09 test-debug  # 显然debug版本占用内存更大

b. 读取ELF格式的可执行程序

[jkl@VMCentos7 dir.gdb]$ readelf -S test-debug | grep -i debug[27] .debug_aranges    PROGBITS         0000000000000000  00004d94[28] .debug_info       PROGBITS         0000000000000000  00004dc4[29] .debug_abbrev     PROGBITS         0000000000000000  00004e8d[30] .debug_line       PROGBITS         0000000000000000  00004f22[31] .debug_str        PROGBITS         0000000000000000  00004f99
# debug版本有关于debug的信息
[jkl@VMCentos7 dir.gdb]$ readelf -S test | grep -i debug  # release版本没有debug信息
[jkl@VMCentos7 dir.gdb]$ 

4. 正式使用gdb 

语法:

gdb [选项] 被调试的可执行程序文件

常见选项:

-q:安静模式,不打印介绍信息和版本信息

默认情况会打印gdb的版本信息。

[jkl@VMCentos7 dir.gdb]$ gdb test-debug
GNU gdb (GDB) Red Hat Enterprise Linux 9.2-7.1.0.4.al8
Copyright (C) 2020 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.
Type "show copying" and "show warranty" for details.
This GDB was configured as "x86_64-redhat-linux-gnu".
Type "show configuration" for configuration details.
For bug reporting instructions, please see:
<http://www.gnu.org/software/gdb/bugs/>.
Find the GDB manual and other documentation resources online at:<http://www.gnu.org/software/gdb/documentation/>.For help, type "help".
Type "apropos word" to search for commands related to "word"...
Reading symbols from test-debug...
(gdb) # 此时已经进入了debug模式

 如果想不打印版本信息,则添加 -q 选项。

[jkl@VMCentos7 dir.gdb]$ gdb -q test-debug
Reading symbols from test-debug...
(gdb) 

查看源码:

list(简写 l): 查看源程序代码,默认显示10行,按回车键继续看余下的。 

运行程序:

 run(简写 r) :运行程序直到遇到 结束或者遇到断点等待下一个命令;

设置断点:

break(简写 b) :格式 b 行号,在某行设置断点;
info breakpoints(简写 i b) :显示断点信息

显示断点信息对应的含义:

  • Num:断点编号
  • Disp:断点执行一次之后是否有效 kep:有效 dis:无效
  • Enb: 当前断点是否有效 y:有效 n:无效
  • Address:内存地址
  • What:位置

单步执行 :

next (简写为 n):逐过程调试会执行下一行,当遇到函数调用时,会一次性执行完该函数,也就是说不会进到函数体内部,相当于VS中的逐过程(F10)。
 
step (简写为 s):单步调试时同样会执行下一行,当遇到函数调用时,会进入改函数体内部,相当于VS中的逐语句(F11)。

查看变量:

print     变量名   查看变量的值

whatis 变量名    查看变量类型

退出gdb:

用quit(简写q)命令退出gdb

指令使用: 

run:简记为 r ,其作用是运行程序,当遇到断点后,程序会在断点处停止运行,等待用户输入下一步的命令。
continue (简写c ):继续执行,到下一个断点处(或运行结束)
next:(简写 n),单步跟踪程序,当遇到函数调用时,也不进入此函数体;此命令同 step 的主要区别是,step 遇到用户自定义的函数,将步进到函数中去运行,而 next 则直接调用函数,不会进入到函数体内。
step (简写s):单步调试如果有函数调用,则进入函数;与命令n不同,n是不进入调用的函数的。
until:当你厌倦了在一个循环体内单步跟踪时,这个命令可以运行程序直到退出循环体。
until+行号: 运行至某行,不仅仅用来跳出循环
finish: 运行程序,直到当前函数完成返回,并打印函数返回时的堆栈地址和返回值及参数值等信息。
call 函数(参数):调用程序中可见的函数,并传递“参数”,如:call gdb_test(55)
quit:简记为 q ,退出gdb

设置断点:

break n (简写b n):在第n行处设置断点。
(可以带上代码路径和代码名称: b 文件名:行号)
b fn1 if a>b:条件断点设置。
break func(break缩写为b):在函数func()的入口处设置断点,如:break cb_button。
delete 断点号n:删除第n个断点。
disable 断点号n:暂停第n个断点。
enable 断点号n:开启第n个断点。
clear 行号n:清除第n行的断点。
info b (info breakpoints) :显示当前程序的断点设置情况。
delete breakpoints:清除所有断点。

查看源码:

list :简记为 l ,其作用就是列出程序的源代码,默认每次显示10行。
list 行号:将显示当前文件以“行号”为中心的前后10行代码,如:list 12
list 函数名:将显示“函数名”所在函数的源代码,如:list main
list :不带参数,将接着上一次 list 命令的,输出下边的内容。

打印:

print 表达式:简记为 p ,其中“表达式”可以是任何当前正在被测试程序的有效表达式,比如当前正在调试C语言的程序,那么“表达式”可以是任何C语言的有效表达式,包括数字,变量甚至是函数调用。
print a:将显示整数 a 的值。
print ++a:将把 a 中的值加1,并显示出来。
print name:将显示字符串 name 的值。
print gdb_test(22):将以整数22作为参数调用 gdb_test() 函数。
print gdb_test(a):将以变量 a 作为参数调用 gdb_test() 函数。
display 表达式:在单步运行时将非常有用,使用display命令设置一个表达式后,它将在每次单步进行指令后,紧接着输出被设置的表达式及值。如: display a。
watch 表达式:设置一个监视点,一旦被监视的“表达式”的值改变,gdb将强行终止正在被调试的程序。如: watch a。
whatis :查询变量或函数。
info function: 查询函数。

查看运行信息:

where/bt :当前运行的堆栈列表。

bt backtrace 显示当前调用堆栈。

up/down 改变堆栈显示的深度。

set args 参数:指定运行时的参数。

show args:查看设置好的参数。

info program: 来查看程序的是否在运行,进程号,被暂停的原因。

分隔窗口:

layout:用于分割窗口,可以一边查看代码,一边测试:

layout src:显示源代码窗口

layout asm:显示反汇编窗口

layout regs:显示源代码/反汇编和CPU寄存器窗口

layout split:显示源代码和反汇编窗口

Ctrl + L:刷新窗口

总结


本篇博客就结束啦,谢谢大家的观看,如果公主少年们有好的建议可以留言喔,谢谢大家啦!

这篇关于【Linux系统编程】第十五弹---调试器gdb使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/973560

相关文章

java并发编程之CyclicBarrier(循环栅栏)

package com.zhong;import java.util.concurrent.CyclicBarrier;/*** Cyclic意思是循环,Barrier意思是屏障,那么CyclicBarrier翻译过来就是循环栅栏。* 它是一个同步辅助类,能让一组线程互相等待,* 直到这一组线程都到了一个公共屏障点,各线程才能继续向下执行。因为该屏障能够在释放等待线程后继续重用,所以叫循环屏障。*

System类获取系统属性

package com.zhong; import java.util.Properties; /**  *   * @author zhong  *  */ public class SystemPropertyTest {          public static void main(String[] args) {         //启动脚本传递参数         if(args

使用Array实现Java堆栈

本教程给出了使用Array 实现Stack数据结构的示例。堆栈提供将新对象放在堆栈上(方法push())并从堆栈中获取对象(方法pop())。堆栈根据后进先出(LIFO)返回对象。请注意,JDK提供了一个默认的Java堆栈实现作为类java.util.Stack。 适用于所有堆栈实现的两个强制操作是: push():数据项放置在堆栈指针指向的位置。pop():从堆栈指针指向的位置删除并返回数据

Spring Boot - 使用类类型信息获取所有已加载的bean

Spring启动会在内部加载大量bean,以最少的配置运行您的应用程序。在这个例子中,我们将学习如何找出所有那些Spring boot加载的bean及其类类型信息。 使用ApplicationContext获取所有已加载的bean 要自动执行方法,当应用程序完全加载时,我正在使用CommandLineRunner接口。CommandLineRunner用于指示bean 在Spring应用程序中

使用ThreadPoolExecutor创建线程池有哪些关键参数

1、ThreadPoolExecutor类的全参数构造方法: /*** Creates a new {@code ThreadPoolExecutor} with the given initial* parameters.** @param corePoolSize the number of threads to keep in the pool, even* if they

Linux命令学习之二

每日一结 ​               命令置换:是将一个命令的输出作为另一个命令的参数,命令格式如下: commond1 `command2` 其中,命令command2的输出将作为命令command1的参数,需要注意的是, 命令置换的单引号为ESC键下方的 ` 其意思就是说,先运行单引号内的命令,再将其命令作为

Linux命令学习之一

每日一结                                                                                        注:当用户不确定一个软件包的类型时, 可使用file命令查看文件类型

Linux文件I/O之一

每日一结(标准I/O)   一  库    本质 :一组实现函数接口 [printf,scanf,strcpy,strlen]   使用标准库好处:屏蔽底层细节,向上层提供统一的接口,提高程序的可移植性   windows 库 : *.dll  linux   库 : *.so    二 系统调用接口    本

嵌入式C和汇编混合编程规则——ATPCS规则

ATPCS规则  为了使单独编译的C语言程序和汇编程序之间能够相互调用,必须为子程序之间的调用规定一定的规则.ATPCS就是ARM程序和THUMB程序中子程序调用的基本规则。 一.ATPCS概述 ATPCS规定了一些子程序之间调用的基本规则.这些基本规则包括子程序调用过程中寄存器的使用规则,数据栈的使用规则,参数的传递规则.为适应一些特定的需要,对这些基本的调用规则进行一些修改得到几种不

Linux网络编程之五

每日一结 组播: ​ 网络地址: 一些特殊的组播地址:   一 组播包收和发 1.发送方 (1)创建用户数据报套接字 (2)填充组播地址和端口  (3)发送数据到组播地址 2.接收方  (1)创建用户数据报套接字 (2)加入组播组  struct ip_mreq mreq; mreq.imr_m