Makefile学习

Makefile概念

Makefile 可以简单的认为是一个工程文件的编译规则,描述了整个工程的编译和链接等规则。其中包含了那些文件需要编译,那些文件不需要编译,那些文件需要先编译,那些文件需要后编译,那些文件需要重建等等。编译整个工程需要涉及到的,在 Makefile 中都可以进行描述。换句话说,Makefile 可以使得我们的项目工程的编译变得自动化,不需要每次都手动输入一堆源文件和参数。

Makefile基本语法

targets : prerequisites
  command

目标(编译的目标或是一个动作):依赖(执行目标所需要依赖的先项)

Tab   命令(具体操作)

注意:我们的目标和依赖文件之间要使用冒号分隔开,命令的开始一定要使用Tab键。

用 Makefile 的方式:首先需要编写好 Makefile 文件,然后在 shell 中执行 make 命令,程序就会自动执行,得到最终的目标文件。

Makefile变量的四种赋值方式

简单赋值 :=

赋值的左侧是立即变量,定义时的赋值立即有效。

输入
boy := I am a boy
girl := $(boy)?
boy := Maybe

qa :
    @echo "girl : $(girl)"
    @echo "boy : $(boy)"

.PHONY : qa

输出
girl : I am a boy?
boy : Maybe


递归赋值 =

赋值的左侧是延时变量,只有被使用时才展开定义。

输入
boy = I am a boy
girl = $(boy)?
boy = Maybe

qa :
    @echo "girl : $(girl)"
    @echo "boy : $(boy)"

.PHONY : qa

输出
girl : Maybe?
boy : Maybe


条件赋值 ?=

赋值的左侧是条件变量,当变量为空时才赋值。

输入
boy := I am a boy
girl := $(boy)?
boy ?= Maybe

qa :
    @echo "girl : $(girl)"
    @echo "boy : $(boy)"

.PHONY : qa

输出
girl : I am a boy?
boy : I am a boy


追加赋值 +=

原变量值之后追加一个新的值,中间用空格隔开。

输入
boy := I am a boy
girl := $(boy)?
boy += Maybe

qa :
    @echo "girl : $(girl)"
    @echo "boy : $(boy)"

.PHONY : qa

输出
girl : I am a boy?
boy : I am a boy Maybe

Makefile模式规则

⽬标的定义中需要包含“%”字符(确切地说是⼀个),包含“%”的⽬标被⽤来匹配⼀个⽂件名,“%”可以匹配任何⾮空字符串。当目标中重现“%”时,目标中“%”所代表的值决定了依赖文件中的“%”的值。

a.o:a.c
    gcc -c a.c
b.o:b.c
    gcc -c b.c

可以写成

%.o:%.c
    gcc -c $<

Makefile伪目标

所谓的伪目标可以这样来理解,它并不会创建目标文件,只是想去执行这个目标下面的命令。

使用伪目标有两点原因:

  • 避免我们的 Makefile 中定义的只执行的命令的目标和工作目录下的实际文件出现名字冲突。
  • 提高执行 make 时的效率,特别是对于一个大型的工程来说,提高编译的效率也是我们所必需的。

第一种情况的使用。如果需要书写这样一个规则,规则所定义的命令不是去创建文件,而是通过 make 命令明确指定它来执行一些特定的命令。

clean:
    rm -rf *.o test

规则中 rm 命令不是创建文件 clean 的命令,而是执行删除任务,删除当前目录下的所有的 .o 结尾和文件名为 test 的文件。当工作目录下不存在以 clean 命令的文件时,在 shell 中输入 make clean 命令,命令 rm -rf *.o test 总会被执行 ,这也是我们期望的结果。如果当前目录下存在文件名为  clean 的文件时情况就会不一样了,当我们在 shell 中执行命令 make clean,由于这个规则没有依赖文件,所以目标被认为是最新的而不去执行规则所定义的命令。因此命令 rm 将不会被执行。为了解决这个问题,删除 clean 文件或者是在 Makefile 中将目标 clean 声明为伪目标。将一个目标声明称伪目标的方法是将它作为特殊的目标.PHONY的依赖。

.PHONY:clean
clean:
    rm -rf *.o test

伪目标的另一种使用的场合是在 make 的并行和递归执行的过程中,此情况下一般会存在一个变量,定义为所有需要 make 的子目录。对多个目录进行 make 的实现,可以在一个规则的命令行中使用 shell 循环来完成。

Makefile函数

1、函数“subst”:完成字符串替换

格式
$(subst <from>, <to>, <text>)


$(subst aaa, AAA, 3a transform 3A aaa)
将字符串“3a transform 3A aaa ”中的“aaa”替换为“AAA”即:“3a transform 3A AAA”

2、函数“patsubst”:完成模式字符串替换

格式
$(patsubst <pattern>, <replacement>, <text>)


$(patsubst %.c, %.o, a.c b.c c.c)
将字符串“a.c b.c c.c”替换为“a.o b.o c.o”

如果text = a.c b.c c.c
那么,“$(text: .c = .o)”等同于“$(patsubst %.c, %.o, $(text))”

3、函数“dir”:获取目录

格式
$(dir <name...>)
 
$(dir </src/a.c>)
提取文件“/src/a.c”的目录部分“/src”

4、函数“notdir”:提取目录名

格式
$(notdir <name...>)
 
$(notdir <src/a.c>)
提取文件“/src/a.c”的非目录部分“a.c”

5、6、函数“foreach”:完成循环;函数“wildcard”:在非规则模式下即变量定义和函数中等同于“%”通配符,将相应对象展开

格式
$(foreach <var>, <list>, <text>)
 
SRCDIRS       := dira dirb dirc 
$(foreach dir, $(SRCDIRS), $(wildcard $(dir) / *.c))
循环将SRCDIRS中的各个目录放进dir变量中,调用wildcard函数提取dir目录下所有.c文件

Makefile自动化变量

自动化变量描述
$@规则中的目标集合,模式规则中,有多个目标的话,”$@“表示匹配模式中定义的目标集合。
$%当目标时函数库的时候表示规则中的目标成员名,如果不是函数库文件,那么其值为空。
$<依赖文件集合中的第一个文件,如果依赖文件时以模式(”%“)定义的,那么”$<“就是符合模式的一系列文件集合。
$?所有比目标新的依赖目标集合,以空格分开。
$^所有依赖文件的集合,以空格分开,如果依赖中有多个重复的文件,”$^“会去除重复的依赖文件,只保留一份。
$+与”$^“类似,但是依赖文件存在重复的话不会去除重复的依赖文件。
$*这个变量表示目标模式中”%“及其之前的部分,如果目标时test/a.test.c,目标模式为a.%.c,那么”$*“就是test/a.test。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

禾末飞雁

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值