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

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

前言

设计样例,先见一下

方案一:

方案二:

在企业里面一般维护软件的源代码的话,要维护几份?

方案一:

方案二:

程序的翻译过程

预处理(进行宏替换)

编译(生成汇编语言)

汇编(生成机器可识别代码)

连接(生成可执行文件或库文件)

函数库

函数库一般分为静态库和动态库两种。

安装一下gcc和g++的静态库

总结


前言

世上有两种耀眼的光芒,一种是正在升起的太阳,一种是正在努力学习编程的你!一个爱学编程的人。各位看官,我衷心的希望这篇博客能对你们有所帮助,同时也希望各位看官能对我的文章给与点评,希望我们能够携手共同促进进步,在编程的道路上越走越远!


提示:以下是本篇文章正文内容,下面案例可供参考

设计样例,先见一下

我们先生成一个可执行程序:

方案一:

我们现在vim中写一段简单的代码。

【Linux编译器-gcc/g++使用】_第1张图片

我们可以用gcc尝试编译一下,看看能否编译过去。

【Linux编译器-gcc/g++使用】_第2张图片

我们先来查看一下gcc的版本。

【Linux编译器-gcc/g++使用】_第3张图片

接下来执行可执行程序。

【Linux编译器-gcc/g++使用】_第4张图片

我们生成的可执行程序也可以自己起一个名字。

【Linux编译器-gcc/g++使用】_第5张图片

方案二:

我们来看一下另一种方法,看看不需要更换gcc的版本号是否可以。

【Linux编译器-gcc/g++使用】_第6张图片

我们来看一下结果。

【Linux编译器-gcc/g++使用】_第7张图片

Linux不会在意文件的后缀,但是gcc/g++这个Linux中的编译器会在意文件的后缀。
gcc不能用来编译c++的文件,gcc是用来编译c语言的、而g++既能编译c语言,也能编译c++。
c++文件的后缀:.cc或.cpp或.cxx。

我们在Xshell中安装一下g++的软件。

【Linux编译器-gcc/g++使用】_第8张图片

在企业里面一般维护软件的源代码的话,要维护几份?

给你一个个人版,再给你一个专业版(个人版的功能没有专业版的功能多),如果按两份来维护的话,比如个人版里面有一个bug,我们把这个bug修正之后,我们的专业版可能也要重新修正;比如专业版里面有bug的话,个人版里面还不一定有,如果有,你也要修正个人版;所以一份有bug的话,我们两个都得看,都得测试,这样太麻烦了,其实我们是可以维护一份源代码的。

那么如何只维护一份源代码?

我们这里引入一个新的方法“条件编译”!

条件编译:它可以根据你指明的编译条件,就可以实现对代码的动态裁剪

我们写一段带有“条件编译”的代码:

【Linux编译器-gcc/g++使用】_第9张图片

将生产的可执行程序运行,查看一下结果。

【Linux编译器-gcc/g++使用】_第10张图片

我们来看一下“条件编译”的使用方法。

方案一:

加入一个宏定义:

【Linux编译器-gcc/g++使用】_第11张图片

上面的#elis V2改成#elif V2,不好意思,上面的写错了!

查看一下结果:

【Linux编译器-gcc/g++使用】_第12张图片

方案二:

【Linux编译器-gcc/g++使用】_第13张图片

上面代码的功能指的是:在软件里面都是某些功能的函数,你想让他有哪些功能,就在对应的代码块里面,把对应函数调用加上。

所以我们只需要维护这一份的代码就可以了,你需要什么样的功能,我们就根据编译条件,对代码进行的动态裁剪。

程序的翻译过程

预处理(进行宏替换)

gcc –E test.c –o test.i

选项“-E”,该选项的作用是让 gcc 在预处理结束后停止编译过程。

选项“-o”是指目标文件,“.i”文件为已经过预处理的C原始程序。

预处理功能主要包括宏定义,文件包含,条件编译,去注释等。

预处理指令是以#号开头的代码行。

编译(生成汇编语言)

gcc –S test.i –o test.s

在这个阶段中,gcc 首先要检查代码的规范性、是否有语法错误等,以确定代码的实际要做的工作,在检查无误后,gcc 把代码翻译成汇编语言。

用户可以使用“-S”选项来进行查看,该选项只进行编译而不进行汇编,生成汇编代码。

【Linux编译器-gcc/g++使用】_第14张图片

汇编(生成机器可识别代码)

gcc –c test.s –o test.o

汇编阶段是把编译阶段生成的“.s”文件转成目标文件

读者在此可使用选项“-c”就可看到汇编代码已转化为“.o”的二进制目标代码了

【Linux编译器-gcc/g++使用】_第15张图片

连接(生成可执行文件或库文件)

gcc test.o –o my.exe

把我们自己写的代码,编译成目标文件,然后和标准库链接,生成可执行程序。

【Linux编译器-gcc/g++使用】_第16张图片

函数库

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

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

函数库一般分为静态库动态库两种。

静态库是指编译链接时,把库文件的代码全部加入到可执行文件中,因此生成的文件比较大,但在运行时也 就不再需要库文件了。其后缀名一般为“.a

动态链接:提前直到目标库的地址。共享动态库,但是一旦动态库缺失,所有的动态链接这个库的程序,都无法执行了!

动态库与之相反,在编译链接时已经知道了需要的库文件中函数的位置,并没有把库文件的代码加入到可执行文件中,而是在程序执行时由运行时链接文件加载库, 这样可以节省系统的开销。动态库一般后缀名为“.so”, 如前面所述的 libc.so.6 就是动态库。gcc 在编译时默认使用动态库。完成了链接之后, gcc 就可以生成可执行文件, 如下所示。gcc test.o –o my.exe

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

【Linux编译器-gcc/g++使用】_第17张图片

举个例子:静态链接:如果内存里有10个可执行程序,比如:每一个可执行程序都要用到printf()函数,那么10个可执行程序,就得拷贝10个printf()函数,所以系统里printf()函数会出现10份。
动态链接:等需要printf()函数方法的时候,只需要跳转到那里就可以了,不需要加载。

我们来验证一下:

【Linux编译器-gcc/g++使用】_第18张图片

【Linux编译器-gcc/g++使用】_第19张图片

安装一下gcc和g++的静态库


总结

好了,本篇博客到这里就结束了,如果有更好的观点,请及时留言,我会认真观看并学习。
不积硅步,无以至千里;不积小流,无以成江海。

你可能感兴趣的:(linux,运维,服务器)