Makefile的基础语法(通配符,假想目标,变量)

1. Makefile的基础语法

1.1 通配符

假如一个目标文件所依赖的依赖文件很多,那样岂不是我们要写很多规则,这显然是不合乎常理的
这时候我们就可以使用通配符来解决这些问题了
这里有一个Makefile文件

test: a.o b.o 
	gcc -o test $^
	
%.o : %.c
	gcc -c -o $@ $<

%.o:表示所用的.o文件
%.c:表示所有的.c文件
$@:表示目标,指的是冒号左边的文件名
$<:表示第1个依赖文件,指的是冒号右边的文件名
$^:表示所有依赖文件,指的冒号是右边所有的依赖文件

1.2 假想目标: .PHONY

当我们想清除文件的时候,我们在Makefile的结尾写:

clean:
	rm *.o test

执行 make :生成第一个可执行文件还有很多依赖文件.o
执行 make clean : 清除所有文件,即执行: rm *.o test。
make后面可以带上目标名,也可以不带,如果不带目标名的话它就想生成第一个规则里面的第一个目标。

  • 但是执行 make clean有一个问题

假如我们现在的路径里有个“clean”文件,这时候运行make,出现以下提示

make: \`clean' is up to date.

现在我们的目录里面有名为“clean”的文件,目标文件是有的,并且没有依赖文件,没有办法判断依赖文件的时间。这种写法会导致:有同名的"clean"文件时,就没有办法执行make clean操作
解决办法:我们需要把目标定义为假象目标,用关键子PHONY

.PHONY: clean //把clean定义为假象目标。他就不会判断名为“clean”的文件是否存在,
	rm *.o test

再运行make clean 就没问题了

1.3 变量

在makefile中有两种变量:
1. 即时变量
对于即使变量使用 “:=” 表示,它的值在定义的时候已经被确定了

2. 延时变量
对于延时变量使用“=”表示。它只有在使用到的时候才确定,在定义/等于时并没有确定下来。

想使用变量的时候使用“$”来引用,如果不想看到命令是,可以在命令的前面加上"@"符号,就不会显示命令本身。当我们执行make命令的时候,make这个指令本身,会把整个Makefile读进去,进行全部分析,然后解析里面的变量。常用的变量的定义如下:

:= # 即时变量
= # 延时变量
?= # 延时变量, 如果是第1次定义才起效, 如果在前面该变量已定义则忽略这句
\+= # 附加, 它是即时变量还是延时变量取决于前面的定义
?=: # 如果这个变量在前面已经被定义了,这句话就会不会起效果,
  • 举例
A := $(C)
B = $(C)
C = abc

#D = hello
D ?= world

all:
	@echo A = $(A)
	@echo B = $(B)
	@echo D = $(D)

C += 123

执行:make
结果:

A =
B = abc 123
D = world

解析:

  1. A为什么等于空
    A为即使变量,在定义时即确定,由于刚开始C的值为空,所以A的值也为空。
  2. B = abc 123
    B为延时变量,只有使用到时它的值才确定,当执行make时,会解析Makefile里面的所用变量,所以先解析C= abc,然后解析C += 123,此时,C = abc 123
  3. D变量在前面没有定义,所以D的值为world,如果在前面添加D = hello,最后D的值为hello。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值