本文主要讲解编译的流程以及动静态库知识,以及项目自动化构建工具make/makefile
gcc/g++
是一个编译器。
我们程序的翻译有四个步骤
.obj
结尾),不可以被执行的,bin.obj)----只是把我们自己的代码进行翻译形成二进制目标文件.obj
文件和库文件某种合并**,形成可执行程序)我们怎么编译文件
我们在编译的时候,使用
gcc +文件名
,就会形成一个可执行程序->a.out
。
如果我们想形成自己想要的名字的可执行程序,我们可以用-o选项,也就是
gcc myfile.c o myfile
,主要是记住-o
后面直接跟我们要的可执行程序的名字。
预处理的作用:
头文件展开,条件编译,宏替换,去注释.
怎么看到预处理的操作
gcc -E
这个选项表示,从现在开始进行程序的翻译,预处理做完,就停下来。但是我们直接打印在屏幕上就太麻烦了,我们将它形成一个文件,就在使用-o
选项,所以最后的操作就是gcc -E myfile.c -o myfile.i
,就可以形成一个myfile.i
的文件,因为一般进行完预处理之后的文件就是用.i
进行后缀。
我们发现这个代码多出来800多行的代码,这个东西就是我们的头文件展开的内容,然后将内容复制到预处理的文件中。
所以这一步叫做头文件的展开
注:打开两个文件的操作,我在介绍vim的文章里面说了,在低行模式下输入vs+文件名
即可
我们发现注释没有了,这一步的操作就是去注释。
我们上面定义过
hehe
,所以这一步就是条件编译
因为上面定义M是100,所以我们在进行预处理操作的时候,直接将M换成100,这个操作就是宏定义
编译的作用
形成的汇编语言
操作
-S
:从现在开始进行程序的翻译,当编译做完,就停下来。
gcc -S myfile.i -o myfile.s
,默认形成myfile.s
的文件
汇编的作用:
汇编代码->可重定位目标二进制文件,不可以被执行的,
就是将我们上一步的汇编代码翻译为二进制文件(以.obj
结尾的文件)
只是把我们自己的代码进行翻译形成二进制文件目标,就是形成
.obj文件
操作:gcc -c myfile.s -o myfile.o
,默认形成.o
文件。
打开文件,就是一堆乱码。
-c,从现在开始进行程序的翻译,当汇编做完,就停下来
链接的作用
将我们这自己形成的.obj文件和库文件
某种合并,形成可执行程序
可以直接进行链接,看下面操作
gcc myfile.o -o myfile
,就形成了可执行程序myfile
。
注:使用./myfile
即可对该执行该程序
我们的选项合起来就是
ESc
,就是前三个操作的选项,我们只需要记住,前两个字母是大写就可以了。
我们三个操作形成的文件后缀是 ,
.iso
。一些镜像文件就是iso
,我们不是ios
,记住千万不要记混。
新命令
ldd+可执行程序
,检测这个可执行程序在当前被形成的时候,都依赖那些库
我们为什们能够在我们 linux下,进行 c的编写呢?
linux系统默认携带了语言级别的 头文件和 语言对应的库!
库分两种-----库本质就是文件
1.静态库:libXXXXX.a
(专门让编译器,对用户的程序进行静态链接的)
2.动态库:lib.XXXX.so
(专门让编译器,对用户的册程序进行动态链接的)
文件的名字怎么看呢?
前缀lib
XXXX后缀.a
我们去掉前缀,去掉后缀,剩下的就是我们的名字。
静态库和静态链接
链接的时候,如果是静态链接,找到静态库,拷贝静态库中的我们所需要的代码到我自己的可执行程序中。
静态链接成功:我们的程序,不依赖任何库,自己就可以独自运行
静态库的缺点:
因为自身拷贝的问题,比较浪费空间
动态库和动态链接
链接的时候,如果是动态链接,找到动态库,拷贝动态库中的我们所需要的代码的地址到我自己的可执行程序中。
动态链接成功:我们的程序,还是依赖库,一但动态库缺失,我们的程序就无法运行。
动态库的优点:
因为可以做到可以被大家共享,所以真正的实现永远都是在库中,程序内部只有地址,比较节省空间。
总结:
动态链接
静态链接,我们先用这个命令创建静态链接gcc myfile.c -o myfile-static -static
make
:是一个命令
makefile
:是一个文件,这个文件在当前的源代码路径下。
make是命令工具,makefile是一个文件,两个搭配使用,完成项目自动化构建。
怎么自动化编译代码
1.快速写一个 demo
。
2.规则
(1)快速写一个demo
先创建一个makefile文件
(2)细节
makefile
是一个围绕依赖关系和依赖方法构建的一个自动化编译关系
完成一件事情,必须要有正确的依赖关系和依赖方法
我们上面makefile的内容如下图
依赖关系
testfile:test.c
这个就是依赖关系,根据test.c
依赖文件 ,形成testfile
目标文件。形成testfile
依赖于test.c
依赖方法
gcc test.c -o testfile
,根据这个依赖方法,形成目标文件。
注意
1.目标文件对应的依赖文件列表可以是空
2.makefile
在进行从上到下的扫描的,只会进行最开始的目标文件。第二个或者之后的并不会直接实现的
.PHONY作用
总是被执行的。
被它修饰的东西就叫做 伪目标
所以我们可以发现,在上面这个makefile
,我们的clean
被他修饰,所以我们的clean
可以一直执行,但是在我们形成目标文件,没有被他修饰,在执行完一次后,目标文件是最新的,我们就不能在进行执行了。如果进行修改,就还可以在执行。
我们知道了上面的知识,我提一个问题,系统怎么知道我们形成目标文件是最新的?
就是看时间,系统会比较源代码的时间和可执行程序的时间。
如果正常的话,我们的可执行程序的时间是比源代码的时间快,所以我们就不能执行了。
但是如果我们修改了,我们就又可以执行了