<Linux> 编译器与调试器—gcc/g++/gdb 的使用

news2024/11/22 10:01:13

< Linux > 编译器与调试器—gcc/g++/gdb 的使用

在这里插入图片描述

文章目录

  • < Linux > 编译器与调试器—gcc/g++/gdb 的使用
  • 一、Linux编译器 - gcc/g++ 使用
    • 1. 编译程序的四个过程
      • 背景知识
      • 预处理
      • 编译
      • 汇编
      • 链接
    • 2. 链接方式与函数库
      • 2.1 动态链接与静态链接
      • 2.2.函数库
      • 2.3.动态库与静态库
    • 3. gcc/g++ 的使用
  • 二、Linux调试器 - gdb 使用
    • 1. debug 和 release
    • 2. gdb 的使用
      • gdb命令汇总

一、Linux编译器 - gcc/g++ 使用

1. 编译程序的四个过程

背景知识

我们知道程序最终需要转化为机器语言(二进制),因为计算机只认识二进制。在程序转化为二进制(编译)的过程中需要经过如下四个过程:

  1. 预处理(进行宏替换)
  2. 编译(生成汇编)
  3. 汇编(生成机器可识别代码)
  4. 连接(生成可执行文件或库文件)

这方面的内容我们在【C】程序环境和预处理已经初步了解过了,但是并没有具体分析过程,接下来我们来演示一下各个过程。

预处理

gcc -E test.c -o test.i
  • 所有的#define删除,并展开所有的宏定义

  • 处理所有的预编译指令,例如:#if,#elif,#else,#endif等。

  • 处理#include预编译指令,将被包含的文件插入到预编译指令的位置。

  • 添加行号信息、文件名标识,便于调试。

  • 删除所有的注释

  • 保留所有的#pragma编译指令,因为在编写程序的时候,我们经常要用到#pargma指令来设定编译的状态或者是指示编译器完成一些特定的动作。

  • 生成.i文件(包括去注释、宏替换、头文件展开、条件编译),编译生成的.i文件不包含任何宏定义,因为宏已经被展开,并且包含的文件已经被插入到.i文件中。

  • -E:从现在开始给我进行程序的翻译,当gcc在预处理完成,就停下来

  • -o:把预处理后的结果写到临时文件test.i里头。“.i”文件为已经过预处理的C原始程序

代码演示

image-20221110172225627

编译

gcc -S test.i -o test.s
  • 扫描、语法分析、语义分析、源代码分析、目标代码生成、目标代码优化。

  • 生成汇编代码。

  • 汇总符号。

  • 生成.s文件。

  • -S:从现在开始进行程序的翻译,当我们编译完成之后,就停下来
  • -o:把编译后的文件写到临时文件test.s里头。

image-20221110172815404

汇编

gcc -c test.s -o test.o
  • 根据汇编指令和特定平台,把汇编指令翻译成二进制形式

  • 合并各个section,合并符号表

  • 将汇编语言翻译成为可重定位二进制文件(.o / obj)
  • -c:从现在开始进行程序的翻译,当我们汇编结束之后,就停下来

image-20221110173227881

我们可以使用 od test.o 指令,将test.o转化为二进制数字。

image-20221110173353192

汇编语言并不能被计算机直接执行,注意这里虽然是二进制文件了,但是仍然不是可执行程序,因为还少了最后一步的链接。

链接

gcc test.o -o mytest
  • 合并各个.obj文件的section,合并符号表,进行符号解析。

  • 符号地址重定位。

  • 生成可执行文件。

  • 这一步就是链接我们自己的程序和库,形成可执行程序!

  • gcc后面不带E/S/c这三个指令,默认就是直接生成可执行程序,省略前三个步骤。

image-20221110174050602

2. 链接方式与函数库

2.1 动态链接与静态链接

为了帮助大家理解,这里我们将一个故事:

image-20221111173628448

假设你上高中了,现在有一个事情你爸给你说一下,就你去学校的时候千万不要去受你那些不好的学长影响,好好学习。你说,好的,满口答应,但是你都不听你爸爸的话。假如这是你们的学校,比如说叫做某某一中,这是学校里面的有一栋宿舍楼,你们的宿舍在501,那么在你进入高校之前,进入,进入这个中学之前你就知道,这个学校管理的非常严格,这个学校它是一个封闭式管理的学校,就是我一到学校里面就出不去了,但是星期天还可以出学校,关键是这个学校禁止你个人携带任何电子设备,这难不倒你,为什么?我们学校附近有一所网吧,叫红树林网吧。你星期天列了一个计划,然后执行。看了一会语文,然后又看了一会英语,然后又看了一会数学,然后去了网吧,愉快的在这上网,那么上网期间,你很愉快的度过了你上网的一个到两个小时,之后就直接回到学校,然后继续做我的计划,然后继续向后再进行执行。你列的这一张清单,那么这个清单就是你写的这个计划,你列的你要做什么,那么这就是你的程序,程序从上到下,全部依次去执行,其中这个程序当中,你要在做的事情,有一部分是靠你自己一个人就可以做的,比如说你想看什么语文英语数学,但是上网的话,你没有上网的工具所以在这里,在你只能跑到到外部去上网,那么其中这个小蚂蚁网吧就叫做库,那么所以当你的程序执行到一定的位置,他会发现,这个方法我这里没有。我要执行函数功能,函数没有,怎么办?我跑到库里面去执行,然后完成之后再把我们的结果再返回,返回之后,然后此时再继续执行我后续的程序。那么这个格式,我们就叫做库函数跳转,也就是我们之间跳转到库的中去执行我们的库函数,那么你怎么知道程序?那么其实说白了,程序背后就是人。那么程序背后怎么知道库在哪里?当年你打电话给你的学长,问他哪里能上网,那么,其中你和你的学长电话的这个过程就叫做,和库建立连接好,那么你的学长在这里他相当于什么?你的学长就相当于一个编译器内部的一个编译器内部的一个链接器,他告诉你的库在哪里,你将来想去调用库方法,那你就去什么地方调就可以了。

image-20221111174240149

但是最近发生一些变故,因为这个网吧的存在,你们附近的学校的学生经常去上网,极大的影响了你们这些学校的成绩,所以你们这几所学校的校长,一起联合举报了这所网吧,说它天天毒害我们学校的青少年,所以警察同志以非法经营的接口把这所网吧给封了。你父母听说了你学校附近的网吧被封了,他们不担心你上网,就是担心你不能上网。于是你爸大手一挥,把网吧的电脑买下来了,并且和学校商量一下,允许你在周末的时候使用电脑,于是在你的宿舍装了一台电脑。这时候你就可以直接在宿舍上网了。这就不用在路上来回跑。

1、什么是静态链接?

静态链接(Static Linking)是由链接器在链接时将库的内容加入到可执行程序中的做法。链接器是一个独立程序,将一个或多个库或目标文件(先前由编译器或汇编器生成)链接到一块生成可执行程序。这里的库指的是静态链接库,Windows下以.lib为后缀,Linux下以.a为后缀。

2、什么是动态链接?

动态链接(Dynamic Linking),把链接这个过程推迟到了运行时再进行,在可执行文件装载时或运行时,由操作系统的装载程序加载库。这里的库指的是动态链接库,Windows下以.dll为后缀,Linux下以.so为后缀。值得一提的是,在Windows下的动态链接也可以用到.lib为后缀的文件,但这里的.lib文件叫做导入库,是由.dll文件生成的。

转载自: 静态链接与动态链接的区别

2.2.函数库

我们的C程序中,并没有定义“printf”的函数实现,且在预编译中包含的“stdio.h”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实“printf”函数的呢?

最后的答案是:系统把这些函数实现都被做到名为 libc.so.6 的库文件中去了,在没有特别指定时,gcc 会到系统默认的搜索路径“/usr/lib”下进行查找,也就是链接到 libc.so.6 库函数中去,这样就能实现函数“printf”了,而这也就是链接的作用

函数库一般分为静态库和动态库两种,下面展开讨论:

2.3.动态库与静态库

  • 头文件:给我们提供了使用的方法,所有的开发环境,具有语法提示,本质是通过头文件帮我们搜索的。(.h)
  • 库文件:给我们提供了可以使用的方法的实现,以供链接,形成我们自己的可执行程序。(.c、.cpp)

静态库:

  • 后缀名Linux(.a),windows(.lib)
  • 静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。静态链接拷贝的并不是.so内部的代码。而是系统必须存在.so结尾的库文件。

动态库:

  • 后缀名Linux(.so),windows(.dll)
  • 动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为“.so”,如前面所述的 libc.so.6 就是动态库。gcc 在编译时默认使用动态库。完成了链接之后,gcc 就可以生成可执行文件,如下所示。 gcchello.o –o hello

静态库和动态库的名称是去掉前缀和后缀剩下的内容。

接下来对比下动静态库的优缺点:

动态链接静态链接
优点大家共享一个库,可以节省资源不依赖任何库,程序可以独立执行
缺点一旦库缺失,会导致几乎所有的程序失效!浪费资源

gcc默认生成的二进制程序是动态链接的,可以通过file命令验证:

image-20221111191813577

如果你想强制改成静态链接,只需要在后面加个-static即可,如下:

image-20221111192834203

如果执行不了静态链接,则说明没有链接到静态库,需要手动执行下面的命令:

gcc(C语言)sudo yum install -y glibc-static
g++(C++)sudo yum install -y libstdc++-static

3. gcc/g++ 的使用

常用选项如下:

  1. -E 只激活预处理,这个不生成文件,你需要把它重定向到一个输出文件里面
  2. -S 编译到汇编语言不进行汇编和链接
  3. -c 编译到目标代码
  4. -o 文件输出到文件
  5. -static 此选项对生成的文件采用静态链接
  6. -g 生成调试信息。GNU 调试器可利用该信息。
  7. -shared 此选项将尽量使用动态库,所以生成文件比较小,但是需要系统由动态库.
  8. -O0 / -O1 / -O2 / -O3 编译器的优化选项的4个级别,-O0表示没有优化,-O1为缺省值,-O3优化级别最高
  9. -w 不生成任何警告信息。
  10. -Wall 生成所有警告信息。

二、Linux调试器 - gdb 使用

1. debug 和 release

Debug:Debug 通常称为调试版本,通过一系列编译选项的配合,编译的结果通常包含调试信息,而且不做任何优化,以为开发人员提供 强大的应用程序调试能力。

Release:Release通常称为 发布版本,是为用户使用的,一般客户不允许在发布版本上进行调试。所以不保存调试信息,同时,它往往进行了各种优化,以期达到代码最小和速度最优。为用户的使用提供便利。

2. gdb 的使用

  • gdb的安装

在Linux下,安装gdb的命令为:

sudo yum install -y gdb

image-20221111214205425

程序的发布方式有两种,debug模式和release模式

Linux gcc/g++出来的二进制程序,默认是release模式

image-20221111213052149

  • 要使用gdb调试,必须在源代码生成二进制程序的时候, 加上 -g 选项
gcc mytest.c -o mytest_g -g

image-20221111213020386

并且这里很明显能看出debug版本的大小是比release版本的大的,而大出来的部分就是增加的调试信息。可以通过readelf -S这条指令看出来:

image-20221111212941026

下面就开始讲解调试的选项指令:

gdb命令汇总

  • gdb binFile :开始调试

  • list (l)行号:显示binFile源代码,接着上次的位置往下列,每次列10行。

  • list 函数名:列出某个函数的源代码。

  • run(r):运行程序(逐语句)。

  • next(n):单条执行(逐过程)。

  • step(s):进入函数调用

  • break(b) 行号:在某一行设置断点

  • break 函数名:在某个函数开头设置断点

  • info b :查看断点信息。

  • finish(fin):执行到当前函数返回,然后挺下来等待命令

  • print(p):打印表达式的值,通过表达式可以修改变量的值或者调用函数

  • p 变量:打印变量值。

  • set var:修改变量的值

  • continue(c):从当前位置开始连续而非单步执行程序

  • run(r):从开始连续而非单步执行程序

  • d :删除所有断点

  • d 行号:删除序号为n的断点

  • disable 编号:禁用断点

  • enable 编号:启用断点

  • info(i) breakpoints:查看当前设置了哪些断点

  • display 变量名:跟踪查看一个变量,每次停下来都显示它的值

  • undisplay:取消对先前设置的那些变量的跟踪

  • until X行号:跳至X行

  • breaktrace(bt):查看各级函数调用及参数

  • info(i)locals:查看当前栈帧局部变量的值

  • quit(q):退出gdb(ctrl + d )

接下来,我们演示一些常用的指令:

显示代码:l 行数

image-20221112192551409

b 行数(打断点):

image-20221112192608271

info b(查看断点):

image-20221112192638501

d + 断点编号(取消断点):

image-20221112192656979

r (调试运行,在断点处停下):

image-20221112193156335

n(逐语句):

image-20221112192723371

s(逐过程):

image-20221112192735035

c(运行至下一个断点):

image-20221112193142565

fin(直接运行到函数跑完):

image-20221112192815621

bt(函数调用堆栈):

image-20221112192753675

display/undisplay(常显示变量,类似于监视):

image-20221112193751269

until(调整指定行):

image-20221112193046638

p 变量(打印变量值)set var(修改变量的值):

image-20221112192959081

info local(查看当前栈帧局部变量的值):

image-20221112192859170

disable breakpoints(禁用断点):

image-20221112193346188

剩下还有一些指令就留给大家去练习一下了,上面的指令平时用的比较多,希望大家多加练习,能够掌握gdb基本的使用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/5235.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

干货分享:谷歌主动搜索开发客户的万能公式

大家在用谷歌(Google)开发是不是也有以下常见问题&#xff1a; 关键词不准&#xff0c;搜索到的都是零售商&#xff0c;B2C平台搜索到的客户太大&#xff0c;已经被开发多次&#xff0c;开发信不回复搜索到的客户找不到邮箱搜索到的客户与工厂不匹配&#xff0c;无法合作 其实…

车路协同 智能路侧设备网络安全接入技术要求

1 范围 本文件包含智能路侧设备网络安全接入技术要求&#xff0c;包括智能路侧设备连接要求、接入要求、证书管理要求。 本文件适用于智能路侧设备网络安全接入的设计与开发。 2 规范性引用文件 下列文件中的内容通过文中的规范性引用而构成本文件必不可少的条款。其中&…

Simulink永磁同步电机控制仿真:过调制及电流重构

在一些高功率密度的应用场景中&#xff0c;追求极致的电压利用率&#xff0c;这个时候要用到过调制技术&#xff1b;当svpwm工作在过调制区域时&#xff0c;逆变电桥会在一个基波周期内多次达到100%占空比&#xff0c;且较多时间处于较高的占空比&#xff0c;这个时候下桥臂电流…

数据结构系列学习(七) - 链栈(Chain_Stack)

目录 引言&#xff1a; 学习&#xff1a; 代码实现&#xff1a; 头文件&#xff08;Chain_Stack.h&#xff09;&#xff1a; 设置链栈中的元素范型&#xff1a; 链栈的结构体设计&#xff1a; 所有功能函数的声明&#xff1a; 源文件&#xff08;Chain_Stack.cpp&#…

性能测试场景设计之 阶梯性能场景(负载测试场景)

「负载测试&#xff1a;」 逐步增加并发用户数。看服务器的最大拐点区间在哪里。再缩小拐点区间&#xff0c;找出最大并发用户数。 使用方式&#xff1a; 安装 jpgc插件 添加线程组 每次递增10个并发 This group will start&#xff1a;给定当前负载的并发用户数First, wait …

机器学习-集成算法

文章目录集成算法1. 定义2. 具体模型2.1. Bagging2.2. Boosting2.3. Stacking3. 随机森林3.1. 树模型结构3.2. 随机森林的优点3.3. 分类与回归问题3.4. 树模型个数问题3.5. 参数问题(特征重要性)3.6. 可视化展示问题4. 集成基本思想4.1. 硬投票策略步骤4.2. 软投票策略步骤5. B…

【SpringMVC】基础、环境搭建、注解搭建、 and so on……

文章目录SpringMVC【原理】&#xff1a;一、why?&#xff1a;二、环境搭建&#xff08;配置方式&#xff09;&#xff1a;▶1.导入jar包▶2.创建简单测试----创建Controller类继承AbstractController▶3.如何读取Controller文件&#xff0c;创建SpringMVC.xml配置文件&#xf…

使用SPACEDESK时iPad显示Connected-Display OFF解决方法

使用SPACEDESK时iPad显示Connected-Display OFF1. SPACEDESK 概述2. 使用SPACEDESK时iPad显示Connected-Display OFF解决方法结束语1. SPACEDESK 概述 SPACEDESK 是一款网络显示多显示器软件&#xff0c;适用于电脑桌面复制&#xff08;镜像或克隆&#xff09;&#xff0c;能将…

[附源码]java毕业设计代驾服务系统

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

初识C语言

初识C语言什么是C语言&#xff1a;环境配置&#xff08;VS2019介绍&#xff09;&#xff1a;你的第一个C语言程序常量与变量变量的分类局部变量与全局变量的对比变量的使用&#xff1a;变量的作用域和生命周期&#xff08;重要&#xff09;生命周期常量字符串转义字符注释转义字…

章节1 计算机体系结构

1.2.1-计算机硬件组成-CPU 计算机组成 台式机硬件-内部 台式机硬件-外部结构 CPU Center Processing Unit&#xff08;中央处理器/处理器&#xff09; 常见的电脑处理器&#xff1a; Inetl奔腾8086、酷睿i5 i7 i9&#xff1b;AMD 锐龙 常见的手机处理器&#xff1a; 高通…

SOT-23和SOT-223三极管及MOS管封装区别以及示意图

SOT23封装 三极管 三极管为SOT23封装时&#xff0c;无论是NPN还是PNP。 引脚分布1脚为基极(B)&#xff0c;2脚为发射极(E)&#xff0c;3脚为集电极©。 NPN PNP MOS管 MOS管为SOT23封装时&#xff0c;无论是N-MOS还是P-MOS。 引脚分布1脚为栅极(G)&#xff0c;2脚为源极(S…

Vue3树形控件实现跳转页面

今天想分享的是Vue3中利用element-plus中的树形控件实现跳转页面&#xff0c;做成类似标签页tabs一样的效果。 1、什么是树形控件 树形控件即可以展现层级结构&#xff0c;可以清晰的看到什么需求在哪个项目下&#xff0c;如下图所示 树形控件直通车 2、分析树形控件的基本…

cspj2022 T4 上升点列(point)题解(floyd)

样例一&#xff1a; 8 2 3 1 3 2 3 3 3 6 1 2 2 2 5 5 5 3 样例一输出&#xff1a; 8 样例二&#xff1a; 4 100 10 10 15 25 20 20 30 30 样例二输出&#xff1a; 103 一、题目解析&#xff1a; 平面上有若干个点&#xff0c;若点[i]可以沿着x或y增加方向移动达到点[j…

树的存储结构

双亲存储结构 ● 做法 • 一种顺序存储结构 , 用一组连续空间存储树的所有节点, • 同时在每个节点中附设一个伪指针指示其双亲节点的位置 • 按顺序依次存储 , 逐层逐个节点存储在数组里面 ● 定义存储结构 typedef struct {//定义数据区ElemType data;//指向其双亲在数组中的…

对象的比较(上)PriorityQueue中的底层源码解析

作者&#xff1a;~小明学编程 文章专栏&#xff1a;Java数据结构 格言&#xff1a;目之所及皆为回忆&#xff0c;心之所想皆为过往 目录 问题引入 offer() 扩容 构造方法 grow() siftUp() siftUpComparable&#xff08;&#xff09; 问题引入 问题是这样的&#xff0c…

LeetCode ——160. 相交链表,142. 环形链表 II

✅<1>主页&#xff1a;C语言的前男友 &#x1f4c3;<2>知识讲解&#xff1a;LeetCode经典链表笔试题目 &#x1f525;<3>创作者&#xff1a;C语言的前男友 ☂️<4>开发环境&#xff1a;Visual Studio 2022 &#x1f3e1;<5>系统环境&#xff1a;…

机器学习笔记 十八:基于3种方法的随机森林模型分析房屋参数重要性

这里写自定义目录标题1. 探索性数据分析1.1 数据集分割&#xff08;训练集、测试集&#xff09;1.2 模型拟合2. 特征重要性比较2.1 Gini Importance2.2 Permutation Importance2.3 Boruta3. 特征比较3.1 Gini Importance3.2 Permutation Importance3.3 Boruta4. 模型比较将机器…

算法设计与分析复习(一)

判断题&#xff1a; 如果一个NP完全问题能在多项式时间内得到解决&#xff0c;那么NP中的每一个问题都能在多项式时间内求解。&#xff08;T&#xff09;可以用如下方法来证明某结论X成立&#xff1a;先假设X不成立&#xff0c;在此假设基础上推出X成立&#xff0c;则可以证明…

8、Bean的循环依赖问题

8.2 singleton下的set注入产生的循环依赖(解决) A对象中有B属性。B对象中有A属性。这就是循环依赖。我依赖你&#xff0c;你也依赖我。 比如&#xff1a;丈夫类Husband&#xff0c;妻子类Wife。Husband中有Wife的引用。Wife中有Husband的引用。 Husband Data public class …