Makefile专题是博主参考陈皓前辈的《跟我一起学Makefile》,特此鸣谢!!!
索引:鼠标右键上拉可回来
1、写在前面的话
最近,在学习汇编的时候,发现这种整理方式挺好的。于是乎,想把之前学习到1/5的Makefile也整理一下,比较有条理性,这样,工作或者学习的时候可以方便些。
这篇博客主要讲下Makefile的基本格式和一些规则。
2、Makefile的功能和初心
01 Makefile的功能
Makefile定义了一系列规则,比如:文件的先后、重新编译等。就像shell脚本,可自动化编译、一旦写好,只需一个make命令,整个功能就能完全自动编译,提高软件的开发效率。02 Makefile的初心
我们使用Makefile有最基本的初心,也就是我们make的时候想要达到的效果,这几个效果可以归结如下几点:(1)如果这个工程没有编译过,那么我们的所有C文件都要编译并被链接(按需)。
(2)如果这个工程某些C文件被修改,那只编译被修改的C文件,并链接目标程序。
(3)如果这个工程的头文件被改变了,那么我们需要编译引用了这几个头文件的C文件,并链接目标程序。
3、Makefile的基本格式
# Makefile的基本格式
target1 ...: prerequisites ...
command # 注意在命令前面需要添加一个Tab键
...
target2 ...: prerequisites ...
command
...
target:目标文件,是命令执行后生成的目标,可以是ObjectFile,也可以是执行文件。还可以是一个标签(Label,类似clean)。
prerequisites:先决条件、依赖,也就是要生成那个target所需要的文件或者是目标。
command:make需要执行的命令(任意的shell命令)。
4、Makefile的核心规则
01 target中一个或多个的目标文件依赖于prerequisites中的文件,其生成规则定义在command中。
02 如果prerequisites中如果有一个以上的文件比target文件要新的话,command所定义的命令就会被执行
5、示例
原文的有些复杂,自己弄了一个,用这个吧!!!(源于B站UP婆主)
CC = gcc # 定义变量,之后可以使用$(CC)来替换gcc
CFLAGS = -lm -Wall -g # 定义变量,之后可以使用$(CFLAGS)来替换-lm -Wall -g
all: main_max main_min # 用all来包含需要编译出来的可执行文件(2个及以上时)
main_max: main_max.c foo.o bar.o # 将编译好的.o文件链接成可执行文件
$(CC) $(CFLAGS) main_max.c foo.o bar.o -o main_max
main_min: main_min.c foo.o bar.o
$(CC) $(CFLAGS) main_min.c foo.o bar.o -o main_min
foo.o: foo.c # 通过编译.c生成需要的.o文件(如果时间更新或者文件首次存在)
$(CC) $(CFLAGS) -c foo.c
bar.o: bar.c
$(CC) $(CFLAGS) -c bar.c
clean: # 清除生成的main_max,main_min和所有的.o文件
rm *.o main_max main_min
解释:
01 有些Makefile中会加入反斜杠(\),这是换行符的作用,使用其便于Makefile的阅读。
02 如果要同时编译出两个可执行文件,需要引入all,如果不加all的话,就会导致只执行了main_max、foo.o、bar.o没有执行main_min。
03 Makefile中注释的格式:# + 内容。
04 在最开始的两行作用类似宏定义,之后command中执行替换的作用。
05 命令前面一定要是Tab键开头,如果 prerequisites 文件的日期要比targets文件的日期要新,或者target不存在的话,那么,make就会执行后续定义的命令(核心规则)。
06 clean不是一个文件,它只不过是一个动作名字,有点像C语言中的lable一样,其冒号后什么也没有,make就不会自动去找文件的依赖性,也就不会自动执行其后所定义的命令。要执行其后的命令,就要在make命令后明显得指出这个lable的名字。这样的方法非常有用,我们可以在一个makefile中定义不用的编译或是和编译无关的命令,比如程序的打包,程序的备份等等。