01:linux基础---gcc/makefile/gdb

news2025/1/18 4:33:27

目录

一:GCC

1:简历

2:编译过程

 3:编译顺序

A:预处理

B:编译        

C:汇编

D:链接

4:编译选项

5:静/动态库

A: 前言

B: 库

C:静态库

D:动态库

6:练习

A: 目录结构

B: 目录结构

二:makefile

1:makefile变量

A:make变量预定义变量

B:make自动化变量

2:3个等号

A:递归定义方式。

B:直接定义方式 :=

C:条件定义法 ?=

四:调式工具GDB

1:启动GDB的2中方式

A: GDB+调式文件的名字

 B: 进入gdb

2:调试命令


一:GCC

1:简历

        编辑c语言或者c++的工具

        GCC (GNU Compiler Collection),即 GNU 编译器套装,是一套由 GNU 开发的编程语言编译器。它是一套以 GPL 及 LGPL 许可证所发行的自由软件,也是 GNU 计划的关键部分还是自由的类 UNIX 及苹果计算机 Mac OS X 操作系统的标准编译器。GCC(特别是其中的C 语言编译器)也常被认为是跨平台编译器的事实标准。
        Linux 系统下的 GCC 编译器实际上是 GNU 编译工具链中的一款软件,可以用它来调用其他不同的工具进行诸如预处理、编译、汇编和链接这样的工作。GCC 不仅功能强大、性能优越,其执行效率比一般的编译器相比要高 20%~30%,而且由于其是 GNU 项目之一,是开源的软件,我们可以直接从网上免费地下载安装它,是名副其实的免费大餐。

2:编译过程

        cpu只能读取二进制文件, 所以我们需要把c或者c++使用GCC编译为二进制文件

格式 gcc [选项] [要编译的文件] [选项] [目标文件]

gcc hello.c - o hello

结论 :

        如果我们直接使用  gcc hello.c  会默认为我们生成一个a.out可执行文件

        如果我们直接使用  gcc -c hello.c 会默认生成一个以本文件为名字的.o文件   (hello.o)

        如果我们直接使用  gcc -c hello.c -o 文件名     会默认生成一个指定的文件名(文件名由我们指定)

 编译顺序 :

 3:编译顺序

A:预处理

        把原本的xxx.c文件通过预处理器 cpp变为xxx.i

        gcc会对.c文件默认进行预处理操作;  GCC在第一个阶段会调用预处理器 cpp 来对 C 源程序进行预处理,所谓的预处理就是解鲁源程序当中所有的预处理指令,那些如#include、 #define、#if等以井号(#)开头的语句就是预处理指令,预处理指令实际上并不是C 语言本身的组成部分,而是为了更好地组织程序所使用的一些“预先处理的”工作,这些工作用一种称为与处理指令的语句来描述,然后用预处理器来解释,这些工作包括我们熟悉的诸如文件包含、宏定义、条件编译等。

B:编译        

        把xxx.i文件通过编译器(ccl)编译为汇编语言.s

        经过预处理之后生成的.i 文件依然是一个文本文件,不能被处理器直接解释,我们需要进一步编译。接下来的编译阶段是 4 个阶段中最为复杂的阶段,它包括词法和语法的分析,最终生成对应硬件平台的汇编语言(不同的处理器有不同的汇编格式),具体生成什么平台的汇编文件取决于所采用的编译器,如果用的是 GCC,那么将会生成x86 格式的汇编文件,如果用的是针对 ARM平台的交叉编译器,那么将会生成 ARM 格式的汇编文件。

C:汇编

        把xxx.s文件通过汇编器 as 编译为二进制文件.o

        编译器 gcc 将会调用汇编器 as 将汇编源程序翻译成为重定位文件。汇编指令跟处理器直接运行的二进制指令流之间基本是一一对应的关系,该段只需要将.s 文件里面的汇编翻译成指令即可

        这个件是一个 ELF 格式的可重定位 (relocatable) 文件。所谓的可重定位,指的是该文件虽然已经包含可以让处理器直接运行的指令流,但是程序中所有的全局符号尚未定位。所谓的全局符号,就是指函数和全局变量,函数和全局变量默认情况下是可以被外部文件引用的,由于定义和调用可以出现在不同的文件当中,因此它们在编译的过程中需要确定其入口地址,如 a.c 文件里面定义了一个函数 func0,b.c 文件里面调用了该函数,那么在完成第 3 阶段汇编之后,b.o 文件里面的函数 func0的地址将是 0,显然这是不能运行的,必须找到 a.c 文件里面函数 func0的确切的入口地址,然后将 b.c 中的“全局符号”func 重新定位为这个地址,程序才能正确运行。因此,接下来需要进行第 4 个阶段: 链接。

D:链接

        经过汇编之后的可重定位的文件不能直接运行,因为还有两个很重要的工没完成,首先是重定位,其次是合并相同权限的段, 标准的c库和内置库是如此的基本, 因此-lc -lgcc 是默认的可以省略

小结:
1)输入文件的后缀名和选项共同决定gcc到底执行那些操作。
2)在编译过程中,除非使用了-E、-S、-c选项(或者编译出错阻止了完整的编译过程)
   否则最后的步骤都是链接。

直接生成可执行文件:

4:编译选项

-w    不生成任何警告信息     

-D   相当于程序中的define

头文件默认在 : 默认路径--编译器中的include中

include <add.h>

<>使用这个引用, 他会在默认的路径include中查找 , 以为add.h 是我们自己写的默认路径中没有,所以直接错误

解决方法: A:把我们写的add.h加入到默认头文件路径include中去

                B:gcc -c hello.c -o hello.o -I后面加头文件的路径

                C: 变为" "---直接在当前目录下找头文件

include "add.h"

""-------会在当前目录下查找头文件

5:静/动态库

A: 前言

函数库的概念

        printf函数在库中; 库的默认路径为:编译器中的lib目录

      库可以指定:

-L选项指定的目录            -l选项指定库(没有空格)

-labc 意思 找libabc.so或者-labc 意思 找libabc.a的库

        我们的C程序中,并没有定义“printf”的函数实现,且在预编译中包含的“stdio.h”中也只有该函数的声明,而没有定义函数的实现,那么,是在哪里实现“printf”函数的呢?
        系统把这些函数实现都被做到名为 libc.so.6 的库文件中去了,在没有特别指定时,gcc 会到系统默认的搜索路径“/usr/lib64”下进行查找,也就是链接到 libc.so.6 库函数中去,这样就能实现函数“printf”了,而这也就是链接的作用

查看可执行文件所依赖的库文件: ldd hello.exe      (ldd后面必须跟的是可执行文件)

查看当前使用的头文件 :  ls /usr/include  ------头文件默认的储存地址

B: 库

一般链接的过程,是有两种方式的:

  • 动态链接——需要动态库
  • 静态链接——需要静态库
  • Linux下的文件名后缀:.so(动态库),.a(静态库)
  • Windows下的文件名后缀:.dll(动态库),.lib(静态库)

所以函数库也就分为静态库和动态库两种:

        静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也就不再需要库文件了。其后缀名一般为“.a”
        动态库与之相反,在编译链接时并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库,这样可以节省系统的开销。动态库一般后缀名为“.so”,如前面所述的 libc.so.6 就是动态库。gcc 在编译时默认使用动态库。完成了链接之后,gcc 就可以生成可执行文件。

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

file hello.exe

 也可以改为静态库(加上--static) :  #不加static的话默认生成运态库

gcc -static hello.c -o hello

C:静态库

编静态库;
    ar -r [lib自定义库名.a] [.o] [.o].....
链接成可执行文件:
    gcc [.c] [.a] -o [自定义输出文件名]
    gcc [.c] -o [自定义输出文件名] -l[库名] -L[库所在路径]

方法2:

D:动态库

第一程序链接时指定链接库的位置,就是使用-wl,-rpath=<link_path>参数,<link_path>就是链接库的路径 ; 上面就是指定了链接的位置在当前目录,这种情况只有在当前目录执行./foo时,才是可以正确使用的          因为在编译链接时并没有把库文件的代码加入到可执行文件中,使用不加-wl, -rpath 运行动态库需要加环境变量

编译二进制.o文件
    gcc -c -fpic [.c/.cpp][.c/.cpp]...
编库
    gcc -shared [.o][.o]...-o [lib自定义库名.so]
连接动态库到可执行文件
    gcc [.c/.cpp] -0 [自定义可执行文件名] -l[库名] -L[库路径] -wl,-rpath=[库路径]

加入环境变量:

export LD_LIBRARY_PATH=/opt/gtk/lib:$LD_LIBRARY_PATH

6:练习

A: 目录结构

直接生成:

创建静态库:
首先创建静态库, 然后在链接为可执行文件

创建动态库:

obj=demo.o add.o

exe:$(obj)
    $(CC)  $^ -o $@
%.o:%.c
    $(CC) -c $^ -o $@
add.o:add.h
clean:
    rm *.o cs-f    //-f 强行删除

改进makefile:

上面的add.o依赖与add.h 需要打开这个文件, 来确定头文件. 我们来改进这个

obj=demo.o add.o
text:$(obj)
	$(CC) $^ -o $@
 # 需要判断是否存在依赖文件
 # .demo.o.d .add.o.d

dep:=$(foreach f,$(obj),.$(f).d)
dep:=$(wildcard $(dep))
#把依赖文件包含进来
ifneq ($(dep),)
	include $(dep)
endif

%.o:%.c
	$(CC) -Wp,-MD,.$@.d  -c $< -o $@

clean:
	rm *.o cs -f 
	
distclean:
	rm $(dep) *.o cs -f 

        -Wp,-MD,    -------生成相应的依赖文件  .--表示他是一个隐藏文件    ls -a 查看隐藏文件

        foreach-----把obj中的每个成员(demo.o add.o)赋值改f, 然后在替换

        wildcard-----展开为已经存在的、使用空格分开的、匹配此模式的所有文件列表。假如不存在任何符合此模式的文件,函数会忽略模式字符并返回空。

ifneq ($(dep),)
    include $(dep)
endif

        # 当修改.h文件之后,判断执行这个include $(dep_files), 表示如果变量 dep 不为空,则包含 dep 文件。这个语句通常用于自动化构建中,用来检查依赖关系并自动更新相关文件。

B: 目录结构

直接编译:

 创建静态库:
首先创建静态库, 然后在链接为可执行文件

创建动态库:  因为上面已经加上了环境变量,所以这次我们不用加环境变量

二:makefile

前言:

        makefile的本质就是对gcc的优化, 避免了每次编译都对文件中的全部文件进行编译. makefile只会对进行改变的文件进行重新编译, 没有动的文件不编译节约了时间

        当我们要编译成千上万个源程序文件时,光靠手工地使用 GCC 工具来达到目的也许就会很没有效率,我们需要一款能够帮助我们自动检查文件的更新情况,自动进行编译的软件GNU make (工程管理器 make 在不同环境有很多版本分支,比如 Qt 下的 qmake,Windows下的nmake 等,下面提到的 make 指的是 Linux 下的 GNUmake)就是这样的一款软件。

        而Makefile,是 make 的配置文件,用来配置运行 make 时的一些相关细节,比如指定编一个工程项目不管是简单还是复杂,每一个源代码子译选项、指定编译环境等。一般而言,目录都会有一个Makefile 来管理,然后一般有个所谓的顶层 Makefile 来统一管理所有的子目Makefile。

工作原理

        make 的工作目的很清楚: 编译那些需要编译的文件,那么究竟哪些文件需要编译呢?

        这个原理也非常简单 : 根据文件的时间戳来进行判断。每个文件都会记录其最改时间,我们只需要对比源文件及其生成的目标文件时间戳,就可以判断它们的新旧关系,从而决定要不编译。比方说我们刚刚修改了 x.c 这个文件,那么它的间戳将会被更新为当前最新的系统时间,这样make通对比就可以知道 x.c 比 x.o 要新,因此在需要使用x.o就会自动重新编译 x.o,这样又会导致 x.o 的时间戳比image 要新,于是 image 也会被自动重新编译,这种递关系会在每一层目标-依赖之间传递

shell命令执行2个条件  :依赖文件比目标文件新,或是 目标文件还没生成

makefile格式:

目标:依赖
    shell命令
//shell命令前面必须是Tab键,否则不能识别shell命令
exe:main.o add.o
    gcc main.o add.o -o exe
main.o:main.c
    gcc -c main.c -o main.o
add.o:add.h
add.o:add.c
    gcc -c add.c -o add.o

 终极目标就是指的是执行make是默认产生的那个文件----exe文件

第一条规则中的目标将被确立为最终的目标

        第一次执行make命令, 全部编译, 下次执行的时候,那个改变执行那个

流程

        (1)找到由终极目标构成的一套规则(第1 行和第2行)。
        (2)如果终极目标及其依赖列表都存在,则判断它们的时间戳关系,只要目标比任何一个依赖文件旧,就会执行其下面的 Shell 命令。
        (3)如果有任何一个依赖文件不存在,或者该依赖文件比该依赖文件的依赖文件要旧则需要执行以该依赖文件为目标的规则的 Shell 命令。
        (4)如果依赖文件都存在并且都最新,但是目标不存在,则执行其下面的 Shell 命令。

寻找:

make命令默认在当前目录下寻找名字为makefile或者Makefile的工程文件,当名字不为这两者之一时,可以使用如下方法指定:

      make -f 文件名

伪目标:

Makefile中把那些没有任何依赖只有执行动作的目标称为“伪目标”

.PHONY : clean
clean:
    rm -f sc *.o 
#.PHONY---将clean目标声明为伪目标

作用: 避免命令与目录下的文件名重复---------如果当前目录下没有名为“clean”的文件,则rm指令会被执行。如果有的话,由于clean没有依赖文件,所以目标被认为是最新的而不去执行rm指令。
在Makefile中加入伪目标声明可防止这种情况发生

1:makefile变量

A:make变量预定义变量

B:make自动化变量

所以现在上面的makefile命令可以改为:

obj=main.o add.o

exe:$(obj)
    $(CC)  $^ -o $@
main.o:main.c
    $(CC) -c $^ -o $@
add.o:add.h
add.o:add.c
    $(CC) -c $< -o $@
 //========================或者================================
obj=main.o add.o

exe:$(obj)
    $(CC)  $^ -o $@
%.o:%.c
    $(CC) -c $^ -o $@
add.o:add.h
clean:
    rm *.o 文件名-f
                      

2个相同的目标,一个有规则,一个没有规则; 他们会合并在一起

2:3个等号

A:递归定义方式。

A =Ilove $(B)
B= China

        此处,在变量 B 出现之前,变量 A 的定义包含了对变量 B 的引用,由于 A 的定方式是所谓的“递归”定义方式,因此当出现S(B)时会对全文件进行搜索,找到 B 的值并代进 A
A =Ilove $(B)
B= China
        递归定义的变量有以下两个缺点。第一,使用此风格的变量定义,可能会由于出现变量中,结果变量A的值是“Ilove China”。的递归定义而导致make陷入到无限的变量展开过程中,最终使make执行失败。

B:直接定义方式 :=

B= China 

A:=l love $(B)
        此处,定义A 时用的是所谓的“直接”定义方式,即如果其定义里出现有对其他变量的A;=I只会在其前面的语句进行搜寻(不包含自己所在的那一行),而不是搜寻整个文件,因此,如果此处将变量 A 和变量 B 的定义交换一个位置

C:条件定义法 ?=

        有时我们需要先判断一个变量是否已经定义了,如果已经定义了则不做操作,如果没有定义方式。定义则再来定义它的值,这时最方便的方法就是采用所谓的条件定义方式:

A=apple

A?=asddasd

在A?=的前面对于A已经定义了, 所以A这个时候等于apple;  否则A等于asddasd

四:调式工具GDB

GDB是GNU发布的一款功能强大的程序调试工具。GDB主要完成下面三个方面的功能
1、启动被调试程序
2、让被调试的程序在指定的位置停住3、当程序被停住时,可以检查程序状态
如变量值)

因为是调式所以在编译的时候加上 -g  编译结果加入调式信息

1:启动GDB的2中方式

A: GDB+调式文件的名字

 B: 进入gdb

        file 调式程序名

2:调试命令

 

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

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

相关文章

springboot+html实现密码重置功能

目录 登录注册&#xff1a; 前端&#xff1a; chnangePssword.html 后端&#xff1a; controller: Mapper层&#xff1a; 逻辑&#xff1a; 登录注册&#xff1a; https://blog.csdn.net/m0_67930426/article/details/133849132 前端&#xff1a; 通过点击忘记密码跳转…

滚雪球学Java(53):从入门到精通:SimpleDateFormat类高深用法,让你的代码更简洁!

咦咦咦&#xff0c;各位小可爱&#xff0c;我是你们的好伙伴——bug菌&#xff0c;今天又来给大家普及Java SE相关知识点了&#xff0c;别躲起来啊&#xff0c;听我讲干货还不快点赞&#xff0c;赞多了我就有动力讲得更嗨啦&#xff01;所以呀&#xff0c;养成先点赞后阅读的好…

通讯协议学习之路:USART协议理论

通讯协议之路主要分为两部分&#xff0c;第一部分从理论上面讲解各类协议的通讯原理以及通讯格式&#xff0c;第二部分从具体运用上讲解各类通讯协议的具体应用方法。 后续文章会同时发表在个人博客(jason1016.club)、CSDN&#xff1b;视频会发布在bilibili(UID:399951374) 一、…

把微信好友不小心删了,有什么办法找回?

常见的重新加回微信好友有以下几种&#xff1a; ①通过微信群找回&#xff0c;看有无共同的群&#xff0c;重新将对方加回来 ②通过朋友圈的动态找回 ③如果是加不久的好友&#xff0c;可以在微信新的朋友功能里重新添加 ④通过共同好友&#xff0c;让朋友圈推荐好友名片找…

一键式AI智能剪辑,轻松处理视频,释放无限创意!“

想象一下&#xff0c;您可以在几秒钟内完成一个复杂的视频剪辑&#xff0c;而无需投入大量的时间和精力。现在&#xff0c;这个梦想已经成为现实&#xff01;我们的新一代AI智能剪辑技术&#xff0c;将使视频处理变得轻松无压力。 第一步&#xff1a;首先进入好简单批量智剪主…

C#上位机序列9: 批量读写+事件广播

1. 读取配置文件及创建变量信息&#xff08;点位名称&#xff0c;地址&#xff0c;数据类型&#xff08;bool/short/int/float/long/double&#xff09;&#xff09; 2. 读任务&写任务,数据有变化时事件广播通知 using HslCommunication; using HslCommunication.Core; usi…

基于YOLOv8模型的水下目标检测系统(PyTorch+Pyside6+YOLOv8模型)

摘要&#xff1a;基于YOLOv8模型的水下目标检测系统可用于日常生活中检测与定位鱼、水母、企鹅、海鹦、鲨鱼、海星、黄貂鱼&#xff0c;利用深度学习算法可实现图片、视频、摄像头等方式的目标检测&#xff0c;另外本系统还支持图片、视频等格式的结果可视化与结果导出。本系统…

34 机器学习(二):数据准备|knn

文章目录 数据准备数据下载数据切割转换器估计器 kNN正常的流程网格多折交叉训练原理讲解距离度量欧式距离(Euclidean Distance)曼哈顿距离(Manhattan Distance)切比雪夫距离 (Chebyshev Distance)还有一些自定义的距离 就请读者自行研究 再识K-近邻算法API选择n邻居的思辨总结…

MySQL分组查询每组最新的一条数据(提供三种实现方法,附带三种方法查询性能比较和分析查询原理)

目录 一、前言二、注意事项三、准备SQL四、错误查询分析4.1、错误原因 五、实现方法5.1、实现方法一&#xff08;使用 LIMIT 查询&#xff09;5.2、实现方法二&#xff08;使用 DISTINCT 查询&#xff09;5.3、实现方法三&#xff08;使用 MAX(id) 查询&#xff0c;只适用于自增…

一篇文章讲明白double、float丢失精度的问题

1.背景 1.10.1 1.2000000000000002 发现上面计算的值竟然和数学计算不一致 2. 问题 计算机是通过二进制计算的&#xff0c;如果我们在二进制的视角来看待上面问题&#xff0c;就很容易发现问题了。 例如&#xff1a;把「0.1」转成二进制的表示&#xff0c;然后还原成十进制&…

846. 树的重心

输入样例 9 1 2 1 7 1 4 2 8 2 5 4 3 3 9 4 6输出样例&#xff1a; 4 分析&#xff1a;因为有n-1条边&#xff0c;所以每个点必然会连接到其他点&#xff0c;不存在孤立点&#xff0c;因此&#xff0c;我们从1-n任意点开始dfs都是可以的&#xff0c;因为无论怎么样&#xff0…

mrRobot

一、信息收集 1.访问地址 没啥信息&#xff0c;尝试扫下目录 2.目录扫描 key1 发现有wp-admin/和robots.txt robots.txt里面还拿到了一个密码字典&#xff0c;猜测是爆破wp的网站账号密码的 3.访问wp-admin/ ┌──(root&#x1f480;kali)-[~/桌面] └─# sort -u fsoci…

Java并发面试题:(六)悲观锁和乐观锁和Java内存模型和CAS原理

悲观锁和乐观锁的区别 什么是悲观锁&#xff1f; 基本上我们理解的操作前对资源加锁&#xff0c;操作完后释放锁。说的都是悲观锁。悲观锁认为所有的资源都是不安全的&#xff0c;随时会被其他线程操作、更改。所以操作资源前一定要加一把锁、防止其他线程访问。 什么是乐观锁&…

基于geojson-vt和canvas的高性能出图

概述 本文介绍基于geojson-vt和canvas&#xff0c;实现node端高性能出图。 效果 实现 1. canvas绘图 import { createCanvas } from canvasconst tileSize 256; const canvas createCanvas(tileSize, tileSize) const ctx canvas.getContext(2d)2. 处理geojson const g…

python二次开发Solidworks:圆形弹簧

目录 1、手动建模 2、python自动建模 1、手动建模 第一步​&#xff1a;草图1&#xff0c;在上视基准面画一个圆心在原点&#xff0c;直径50mm的圆​&#xff1b; 第二步​&#xff1a;草图2&#xff0c;在上视基准面画两条构造线&#xff0c;一条经过原点方向竖直&#xff0…

【jvm】虚拟机栈之局部变量表

目录 一、说明二、代码分析2.1 代码示例2.2 执行javap2.3 jclasslib插件查看 三、对slot的理解3.1 说明3.2 slot索引图3.3 实例方法的局部变量表3.4 long和double类型变量占2个slot 四、slot的重复利用4.1 说明4.2 变量c复用变量b的槽位 五、静态变量与局部变量对比 一、说明 1…

NEFU计算机网络实验一常见网络命令的使用

一、实验目的 1、理解、验证常用网络命令的原理和功能。 2、掌握常用的网络命令使用方法&#xff0c;合理使用相关命令对网络进行管理与维护。 二、实验内容 网络参数查询命令&#xff1a;IPCONFIG 网络测试命令&#xff1a;ping 路由表命令ROUTE 网络端口查询命令&…

chatglm配置

推荐看这个链接&#xff0c;有些问题解决出处https://zhuanlan.zhihu.com/p/643824521 以及这个https://blog.csdn.net/weixin_40547993/article/details/131775275 1.需要pytorch2.0&#xff0c;所以CUDA推荐11.8 ChatGLM2-6B版本要装PYTORCH2.0&#xff0c;而且要2.0.1 &a…

resultMap 和 resultType的用法和区别详解

resultMap 和 resultType的用法和区别详解 《resultMap 和 resultType的用法和区别详解》摘要引言resultType - 用法和映射示例了解resultType示例演示 resultMap - 区别、高级用法和自定义映射规则详解resultType vs. resultMap高级用法示例演示 Mybatis的CRUD操作总结参考资料…

信息保卫战:揭秘迅软DSE护航企业免受泄密之害

随着网络技术的发展&#xff0c;通过网络应用如网盘、网页、邮件、即时通讯工具传输分享文件变得越来越多&#xff0c;这些工具传输速度快&#xff0c;能够将大容量的文档快速传送给他人&#xff0c;在工作中受到许多人的青睐。 然而由这些传输工具引发的泄密事件也不断增多&am…