什么是Makefile
- 一个工程中的源文件不计其数,其按类型、功能、模块分别放在若干个目录中,Makefile文件定义了一系列的规则来指定哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作,因为Makefile文件就像Shell脚本一样,也可以执行操作系统的命令。
- Makefile带来的好处就是“自动化编译”,一旦写好,只需要一个make命令,整个工程完全自动编译,极大的提高嘞软件开发的效率。make是一个命令工具,是一个解释Makefile文件中指令的命令工具,一般来说,大多数的IDE都有这歌命令,比如Delphi的make,Visual C++的nmake,Linux下GNU的make。
Makefile文件命名和规则
文件命名
makefile或者Makefile
Makefile规则
一个Makefile文件中可以有一个或者多个规则
目标…:依赖…
Tab命令(shell命令)
Tab …
- 目标:最终要生成的文件(伪目标除外)
- 依赖:生成目标所需要的文件或是目标
- 命令:通过执行命令对依赖操作生成目标(命令前必须Tab缩进)
Makefile中的其他规则一般都是为第一条规则服务的
Vim Makefile创建Makefile文件
工作原理
- 在命令执行之前,需要检查规则中的依赖是否存在
- 如果存在,执行命令
- 如果不存在,向下检查其他的规则,检查有没有一个规则是用来生成这个依赖的,如果找到了,则执行该规则中的命令
上图中更优!
- 检测更新,在执行规则中的命令时,会比较目标和依赖文件的时间
- 如果依赖的时间比目标的时间晚,需要重新生成目标
- 如果依赖的时间比目标的时间早,目标不需要更新,对应规则中的命令不需要被执行
变量
- 自定义变量
变量名=变量值 var=hello - 预定义变量
AR:归档维护程序的名称,默认值为ar
CC:C编译器的名称,默认值为cc
CXX:C++编译器的名称,默认值为g++
$@:目标的完整名称,自动变量
$<:第一个依赖文件的名称,自动变量
$^:所有的依赖文件,自动变量 - 获取变量的值
$(变量名)
$(var)
app:main.c a.c b.c
gcc -c main.c a.c b.c -o.app
#自动变量只能在规则的命令中使用
app:main.c a.c b.c
$(CC) -c $^ -o $@
模式匹配
#定义变量
src=sub.o add.o mult.o div.o main.o
target=app
$(target):$(src)
$(CC) $(src) -o $(target)
%.o:%.c
$(CC) -c $< -o $@
函数
$(wildcard PATTERN…)
- 功能:获取指定目录下指定类型的文件列表
- 参数:PATTERN指的是某个或多个目录下的对应的某种类型的文件,如果有多个目录,一般使用空格间隔
- 返回:得到的若干个文件的文件列表,文件名之间使用空格间隔
- 示例
$(wildcard .c ./sub/.c)
返回值格式:a.c b.c c.c d.c e.c f.c
$(patsubst pattern,replacement,text)
- 功能:查找text中的单词(单词以“空格”、“Tab”或“回车”“换行”分割)是否符合模式pattern,如果匹配的话,则以reolacement替换
- pattern可以包括通配符’%‘,表示任意长度的子串。如果replacement中也包含’%‘,那么replacement中的这个’%‘将是pattern中的那个%所代表的字串。(可以用’'来转义,以’%‘来表示真实含义的’%'字符)
- 返回:函数返回被替换过后的字符串
- 示例
$(patsubst %.c,%.o,x.c bar.c)
返回值格式:x.o bar.o
clean:
rm $(objs) -f
make clean