[Linux 高并发服务器]Makefile
[Linux 高并发服务器]Makefile
- [Linux 高并发服务器]Makefile
- 什么是Makefile
- 文件命名和规则
- Makefile工作原理
- 一、命令在执行之前,需要先检查依赖是否存在
- 二、检测更新在执行规则中的命令时,会比较目标和依赖文件的时间
- 变量
- 自定义变量
- 预定义变量
- 获取变量的值
- 模式匹配
- 函数
- 函数1
- 函数2
什么是Makefile
Makefile 文件定义了一系列的规则来指定那些文件需要先编译,那些文件需要后编译,哪些文件需要重新编译,甚至于更复杂的操作。
Makefile带来的好处就是自动化编译,一旦写好,只需要一个make命令,整个工程就完全自动编译。make是一个命令工具,解释Makefile文件中指令的命令工具
大多数IDE自带make,Linux下的GNU就也有
文件命名和规则
命名:
makefile或者Makefile
规则:
一个Makefile文件可以有一个或多个规则
目标...:依赖...
命令(shell命令)
...
目标:
最终要生成的文件(伪目标除外)
依赖:
生成目标所需要的文件或是目标
命令:
通过执行命令对依赖操作生成目标(命令前必须TAB缩进)
Makefile 中的其他规则一般都是为第一条规则服务的
app:sub.c add.c main.c
gcc sub.c add.c main.c -o app
Makefile工作原理
一、命令在执行之前,需要先检查依赖是否存在
如果存在执行这个命令
如果不存在,向下检查其他的规则,看有没有一个规则使用来生成这个依赖的
如果找到了,则执行该规则中的命令
app:sub.o add.o main.o
gcc sub.o add.o main.o -o app
sub.o:sub.c
gcc -c sub.c -o sub.o
add.o:add.c
gcc -c add.c -o add.o
main.o:main.c
gcc -c main.c -o main.o
注意,这么写比上一个写法要好,否则单个更新了就要全部重新编译
二、检测更新在执行规则中的命令时,会比较目标和依赖文件的时间
如果依赖的时间比目标时间玩,需要重新生成目标
如果依赖的时间比目标时间早,不需要重新生成目标,对应规则中的命令也不需要被执行
变量
自定义变量
变量名=变量值
var=hello
预定义变量
AR:
归档维护程序的名称,默认值为ar
CC:
C编译器的名称,默认值为cc
CXX:
C++编译器的名称,默认值为g++
下面三个自动变量,只能在规则的命令中使用
$@:
目标的完整名称
$<:
第一个依赖文件的名称
$^:
所有的依赖文件
获取变量的值
$(变量名)
通过上述知识对Makefile文件进行简化
src=sub.o add.o main.o
target=app
$(target):$(src)
$(CC) $(src) -o $(target)
sub.o:sub.c
gcc -c sub.c -o sub.o
add.o:add.c
gcc -c add.c -o add.o
main.o:main.c
gcc -c main.c -o main.o
模式匹配
虽然Makefile通过自定义变量的方式简化了,但是下面的部分还是显得复杂
sub.o:sub.c
gcc -c sub.c -o sub.o
add.o:add.c
gcc -c add.c -o add.o
main.o:main.c
gcc -c main.c -o main.o
我们可以采用模式匹配的方式继续简化Makefile文件
%.o:%.c
$(CC) -c $< -o $@
%
通配符匹配字符串,两个%
匹配的是同一个字符串
src=sub.o add.o main.o
target=app
$(target):$(src)
$(CC) $(src) -o $(target)
%.o:%.c
$(CC) -c $< -o $@
如果第一个规则的依赖没有被生成,那么会通过下一个规则的模式匹配找到生成依赖的规则
函数
Makefile文件通过通配符得到了进一步优化,但是有个问题,如果.o
文件太多了,那么src就会很长,有没有办法解决这点的呢?
我们可以使用函数来对他进行优化
函数1
$(wildcard PATTERN...)
功能:
获取指定目录下指定类型的文件列表
参数:
PATTERN指的是某个或多个目录下的对应的某种类型的文件,如果有多个目录,一般使用空格间隔
返回:
得到若干个文件的文件列表,文件名之间使用空格间隔
$(wildcard *.c ./sub/*.c)
返回值:a.c b.c c.c d.c e.c f.c
优化后的Makefile文件
src=$(wildcard ./*.c)
target=app
$(target):$(src)
$(CC) $(src) -o $(target)
%.o:%.c
$(CC) -c $< -o $@
函数2
$(patsubst <pattern>,<replacement>,<text>)
功能
查找<text>
中的单词(单词以“空格”、“TAB”、“回音“、“换行”分隔)是否符合模式<pattern>
,如果匹配就用<replacement>
替换
<pattern>
可以包括通配符%表示任意长度字符串,可以\来转化转义字符
返回
函数返回替换后的字符串
$(patsubst %.c,%.o,x.c bar.c)
返回值x.o bar.o
src=$(wildcard ./*.c)
objs=$(patsubst %.c, %.o, $(src))
target=app
$(target):$(objs)
$(CC) $(objs) -o $(target)
%.o:%.c
$(CC) -c $< -o $@
clean:
rm $(objs) -f
使用make clean
命令执行删除.o文件的命令