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

目录

前言

1.看一段样例

 2.程序的翻译过程

1.第一个阶段:预处理

2.第二个阶段:编译

3.第三个阶段:汇编

4.第四个阶段:链接

3.程序的编译为什么是这个样子?

4. 关于编译器

5.链接(动静态链接)


前言

1.首先,我们来看一段样例(见一下)

2.见完之后,我们来看一下程序的翻译过程

3.了解一下翻译的过程为什么是这样的?再顺带谈一下编译器。

4.最后,我们要谈一下动静态链接,看一下它们的特点和区别


1.看一段样例

一段C语言代码:

(编译器为gcc)

在这里,可能你的编译器是不支持在循环体内直接定义变量的,因为编译器版本老旧,这时需要加上-std=99,让编译器支持C99。

在执行编译的时候可以加上-o ,在-o后面可以指定生成的可执行程序的名字。


一段C++代码:

注意C++文件的后缀为(.cc/.cpp/.cxx)

(编译器为g++)

如果没有g++,需要下载安装,下面是g++的安装指令:

yum install -y gcc-c++

在这里我们也要加上选项 -std = c++11

查看运行结果:

如果想指定c++可执行程序的名字,同样是使用-o选项

(注意:gcc不能编译c++的代码,但是g++可以编译c语言的代码)


 2.程序的翻译过程

使用C语言代码示例:

1.第一个阶段:预处理

说明:在这个阶段编译器会进行宏替换,去注释,头文件展开,条件编译

请看下面的代码:

接下来我们使用gcc的命令让程序在每个阶段都停下来,第一个选项(-E),在这里产生了临时文件test.i (-E的含义:从现在开始进行程序的编译,预处理完成,就停下)

在这里,我们发现,源代码中一下子多了八百多行代码,这就是因为<stdio.h>头文件展开拷贝到源文件,所以经过了预处理,头文件就没用了。(我们发现预处理阶段确实进行了宏的替换,去注释)

Linux系统中存在<stdio.h>的文件,才能进行此操作

Linux系统中所有的库:

关于条件编译(编译器在预处理阶段会对代码进行动态的裁剪):

        在C语言程序的预处理阶段,条件编译起着非常重要的作用。预处理阶段是在实际的编译之前进行的,主要目的是对源代码进行一些预处理操作,例如宏替换、头文件包含等。条件编译指令会在预处理阶段根据条件判断是否包含或排除特定的代码块,从而影响编译后生成的目标代码。

        使用条件编译指令可以根据不同的条件选择性地包含或排除特定的代码块。这样可以根据编译条件在不同情况下编译不同的代码,提高程序的灵活性和可移植性。

        

        我们在写条件编译的时候,一般会定义宏去调用,这样的方式效率是很低的,在Linux系统中,在指令界面就可以动态的向源代码添加宏,下面是一段示例:

在Linux中,使用gcc编译器时,可以通过 -D选项来定义预处理宏。当你使用 -D后跟一个宏名时,它告诉编译器在编译源代码之前定义该宏,可以选择性地为宏指定一个值。

        在下面这个例子中:我们为源代码中添加了宏 v1= 1 ,运行可执行程序之后,得到了接过条件编译的结果。

接着看v2 = 1


2.第二个阶段:编译

说明:编译阶段是将经过预处理后的源代码翻译成汇编代码的过程。编译器会将C语言源代码翻译成相应的汇编语言代码。

编译指令:-S表示从现在开始程序的编译,编译完成就停下来。

test.s:C语言代码转成的汇编语言代码。


3.第三个阶段:汇编

说明:汇编器会将汇编代码中的指令和数据转换成机器指令,并生成二进制目标文件。 汇编的结果是生成一个目标文件,通常以.o为扩展名。

编译指令:-c表示从现在开始进行程序的汇编,汇编完成就停下来

这个.o文件是不可执行的


4.第四个阶段:链接

说明:链接目标文件和链接库生成可执行程序(二进制的程序)。


3.程序的编译为什么是这个样子?

        在没有c语言的时候,都是用的二进制编程。三体电视剧中的机器上就是这种打孔纸带。有孔的地方信息可以透过去就是(1),没孔的地方就是(0)。但这种二进制是十分不方便的,还容易出错,这时候就发明了汇编语言,但觉得汇编语言还是不方便,后面就有了c语言。

想一想编译器是从哪里来的,二进制编程需要编译器吗?

        二进制当然是不用编译器的,从汇编开始我们就需要编译器了。编译器是非常不容易写的,我们是把c语言直接转成二进制容易,还是先转成汇编,再让汇编转成二进制容易?当然是后者比较容易,我们只需要将c语言转成汇编就行了,因为汇编转成二进制的过程前人已经帮忙做了(站在巨人的肩膀上),所以c编译器的核心工作只需要将c语言转成汇编就行了。

        因为历史选择了从二进制到汇编,从汇编到更高级的语言,从更高级的语言到c语言。我们编译的过程就是一个回溯历史的过程,这样的效率是最高的,所以c语言的编译是这个样子。


4. 关于编译器

先有语言还是语言的编译器

        当然是先有语言,才有编译器。最早的汇编语言的编译器是用二进制语言写的,汇编语言才能转成二进制跑起来。

编译器的自举

编译器的自举(bootstrapping)是指使用已经存在的编译器来编译自身的过程。简单来说,自举是通过一个较早版本的编译器来编译当前版本的编译器,从而生成一个新的、功能相同但可能更高效的编译器。 自举的过程通常涉及以下步骤:

1. **初始阶段**:在自举过程开始之前,需要有一个已经存在的编译器,通常是由其他编译器或者手动编写的汇编语言程序生成的。

2. **第一次编译**:使用初始阶段得到的编译器,将编译器的源代码编译成可执行文件。这个可执行文件就是一个新的、功能相同的编译器。

3. **第一次自举**:使用第一次编译得到的新编译器,将编译器的源代码再次编译成可执行文件。这一次编译的目标是生成一个和第一次编译得到的新编译器完全相同的新编译器。

4. **迭代自举**:重复进行自举过程,使用新生成的编译器来编译自身的源代码,生成更高效的新版本编译器。这个过程可以多次迭代,每次都会产生一个新的编译器版本,直到达到所需的性能和功能。

        通过自举,编译器可以逐步改进和优化自身,生成更高效的版本。自举的过程可以确保编译器的正确性,因为每个新版本的编译器都是由之前版本编译而来,保证了功能的一致性。

        需要注意的是,自举过程的初始阶段需要使用其他编译器或者手动编写的汇编语言程序,以便启动自举过程。自举是编译器开发和维护中重要的技术,它使得编译器可以自我改进和更新,为编程语言的发展提供了基础。


5.链接(动静态链接)

是什么?
我们的程序和库结合的过程,语言是一定要有自己的标准库的。

(安装开发环境:安装语言的标准库+头文件)

标准库通常包括各种数据结构、算法、输入输出、文件操作、网络通信等常用功能,以帮助开发人员更高效地编写程序。

在Linux中,ldd加上可执行程序,就可以查到该执行程序依赖了什么库

为什么要有链接?

1.让开发站在巨人的肩膀上 2.提高开发的效率

怎么做的?

在Linux系统中,分为两种库,动态库:.so  静态库:.a(动态链接,静态链接)

静态链接
特点:在生成可执行文件的时候(链接阶段),把所有需要的函数的二进制代码都包含到可执行文件中去。因此,链接器需要知道参与链接的目标文件需要哪些函数,同时也要知道每个目标文件都能提供什么函数,这样链接器才能知道是不是每个目标文件所需要的函数都能正确地链接。如果某个目标文件需要的函数在参与链接的目标文件中找不到的话,链接器就报错了。目标文件中有两个重要的接口来提供这些信息:一个是符号表,另外一个是重定位表。
优点:在程序发布的时候就不需要的依赖库,也就是不再需要带着库一块发布,程序可以独立执行。
缺点:程序体积会相对大一些。
如果静态库有更新的话,所有可执行文件都得重新链接才能用上新的静态库。
动态链接
特点: 在编译的时候不直接拷贝可执行代码,而是通过记录一系列符号和参数,在程序运行或加载时将这些信息传递给操作系统,操作系统负责将需要的动态库加载到内存中,然后程序在运行到指定的代码时,去共享执行内存中已经加载的动态库可执行代码,最终达到运行时连接的目的。
优点: 多个程序可以共享同一段代码,而不需要在磁盘上存储多个拷贝。
缺点: 由于是运行时加载,可能会影响程序的前期执行性能。

下面是同一个程序在不同链接下的内存对比,很显然静态链接的内存大小是远大于动态链接

 

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++数据结构与算法——字符串

C第二阶段——数据结构和算法&#xff0c;之前学过一点点数据结构&#xff0c;当时是基于Python来学习的&#xff0c;现在基于C查漏补缺&#xff0c;尤其是树的部分。这一部分计划一个月&#xff0c;主要利用代码随想录来学习&#xff0c;刷题使用力扣网站&#xff0c;不定时更…

英文单词-计算:Calculate与Compute的区别是什么

英文单词-计算:Calculate与Compute的区别是什么 compute 源自法语&#xff1b;calculate 源自拉丁语。在使用上&#xff0c;calculate 使用得更为广泛 calculate侧重人的分析&#xff0c;而compute侧重机器的运算。 calculator是“计算器”&#xff0c;而computer是“计算机”…

爬虫之牛刀小试(十一):爬取某东关于手机的评论

首先我们进入华为官方旗舰店&#xff0c;点击Mate60&#xff0c;得到&#xff1a; 找到存放评论的接口网址&#xff1a; 然后使用cookie模拟登录&#xff0c;得到字典筛选出我们想要的内容。 爬取1000条评论 同样可以对任意商品进行操作&#xff0c;得到16款手机共计16000…

Java使用opencsv完成对csv批量操作

文章目录 前言一、maven二、造数三、代码部分1.OpenCsvController2.OpenCsvUtil3.StudentInfo4.CodeToValue 三、效果展示1.download2.upload 总结 前言 csv文件是不同于excel文件的另一种文件&#xff0c;常常以,作为分隔符&#xff0c;本篇将通过JavaBean的形式完成对csv文件…

2024.2.15 模拟实现 RabbitMQ —— 消息持久化

目录 引言 约定存储方式 消息序列化 重点理解 针对 MessageFileManager 单元测试 小结 引言 问题&#xff1a; 关于 Message&#xff08;消息&#xff09;为啥在硬盘上存储&#xff1f; 回答&#xff1a; 消息操作并不涉及到复杂的增删查改消息数量可能会非常多&#xff…

【机器学习笔记】8 决策树

决策树原理 决策树是从训练数据中学习得出一个树状结构的模型。 决策树属于判别模型。 决策树是一种树状结构&#xff0c;通过做出一系列决策&#xff08;选择&#xff09;来对数据进行划分&#xff0c;这类似于针对一系列问题进行选择。决策树的决策过程就是从根节点开始&…

心理辅导|高校心理教育辅导系统|基于Springboot的高校心理教育辅导系统设计与实现(源码+数据库+文档)

高校心理教育辅导系统目录 目录 基于Springboot的高校心理教育辅导系统设计与实现 一、前言 二、系统功能设计 三、系统实现 1、学生功能模块的实现 &#xff08;1&#xff09;学生登录界面 &#xff08;2&#xff09;留言反馈界面 &#xff08;3&#xff09;试卷列表界…

Java实现实现自动化pdf打水印小项目 使用技术pdfbox、Documents4j

文章目录 前言源码获取一、需求说明二、 调研pdf处理工具word处理工具 三、技术栈选择四、功能实现实现效果详细功能介绍详细代码实现项目目录WordUtilsMain类实现部分&#xff1a;第一部分Main类实现部分&#xff1a;第二部分Main类实现部分&#xff1a;第三部分 资料获取 前言…

win7自带截图工具保存失效解决办法

今日发现一台远航技术的win7中自带的截图工具使用时正常&#xff0c;保存图片时没有弹出保存位置的对话窗口&#xff0c;无法正常保存图片。解决方案如下&#xff1a; 1、进入注册表编辑器。开始-搜索程序和文件-输入 regedit 按下回车键&#xff0c;打开注册表&#xff1b; 2、…

DS Wannabe之5-AM Project: DS 30day int prep day18

Q1. What is Levenshtein Algorithm? Levenshtein算法&#xff0c;也称为编辑距离算法&#xff0c;是一种量化两个字符串之间差异的方法。它通过计算将一个字符串转换成另一个字符串所需的最少单字符编辑操作次数来实现。这些编辑操作包括插入、删除和替换字符。Levenshtein距…

【知识图谱--第三讲知识图谱的存储与查询】

知识图谱的存储与查询 基于关系型数据库的知识图谱存储基于原生图数据库的知识图谱存储原生图数据库实现原理浅析 基于关系型数据库的知识图谱存储 基于原生图数据库的知识图谱存储 原生图数据库实现原理浅析

黑群晖一键修复:root、AME、DTS、转码、CPU型号等

食用方法&#xff1a;SSH连接群晖使用临时root权限执行 AME3.x激活补丁 只适用于x86_64的&#xff1a;DSM7.x Advanced Media Extensions (AME)版本3.0.1-2004、3.1.0-3005 激活过程需要下载官方的解码包&#xff0c;过程较慢&#xff0c;耐心等待。。。 DSM7.1和7.2的AME版…