Makefile语法(转)

转自百问网公众号

假如一个目标文件所依赖的依赖文件很多,我们岂不是要写很多规则?这显然不合乎常理。

我们可以使用通配符解决这些问题。我们对上节程序进行修改代码如下:

test: a.o b.o 
gcc -o test $^

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

 

%.o:表示所用的.o文件

%.c:表示所有的.c文件

$@:表示目标

$<:表示第1个依赖文件

$^:表示所有依赖文件

 

在该目录下增加一个c.c文件,代码如下:

#include<stdio.h>
void func_c()
{
printf("This is C\n");
}

在main函数中调用修改的Makefile,修改后的代码如下:

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

执行:

make

结果:

gcc -c -o a.o a.c
gcc -c -o b.o b.c
gcc -c -o c.o c.c
gcc -o test a.o b.o c.o

 

运行:

./test

 

结果:

This is B

This is C


1.我们若清除文件,在Makefile的结尾添加如下代码即可:

clean:
rm *.o test

 

1).执行make:生成第一个可执行文件。

2).执行make clean: 清除所有文件,即执行:rm *.o test。

 

2.使用Makefile

执行:make [目标]

也可以不跟目标名,若无目标默认第一个目标。执行make的时候,会在makefile里面找到第一个目标然后执行下面的指令生成第一个目标。当执行make clean的时候,会在Makefile里面找到clean这个目标,然后执行里面的命令,这个写法有些问题,原因是我们的目录里面没有clean这个文件,这个规则执行的条件成立,它就会执行下面的命令来删除文件。

如果:该目录下面有名为clean文件怎么办呢?

我们在该目录下创建一个名为“clean”的文件,然后重新执行make,make clean,结果(会有下面的提示:):

make: `clean' is upto date.

 

它根本没有执行我们的删除操作,为什么?

我们之前说,一个规则能够执行的条件:

1).目标文件不存在

2).依赖文件比目标新。

 

现在我们的目录里面有名为“clean”的文件,目标文件是有的,并且没有依赖文件,没有办法判断依赖文件的时间。这种写法会导致:有同名的”clean”文件时,没有办法执行make clean操作。

解决办法:把目标定义为假想目标,用关键字PHONY。 

.PHONY: clean //把clean定义为假想目标。它便不会判断名为“clean”的文件是否存在, 然后在Makfile结尾添加.PHONY: clean语句,重新执行:make clean,就会执行删除操作。


在makefile中有两种变量:

· 1)简单变量(即时变量):

A := xxx // A的值即刻确定,在定义时即确定

对于即时变量使用“:=”表示,它的值在定义的时候已被确定。

 

· 2)延时变量

B = xxx // B的值被使用到时才确定

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

 

想使用变量的时候使用“$”来引用,echo的作用是将内容回显,如果不想看到命令时,可以在命令的前面加上”@”符号,则不会显示命令本身。

当我们执行make命令的时候,make这个指令本身,会把整个Makefile读进去,进行全部分析,然后解析里面的变量。常用的变量的定义如下:

:= // 即时变量

= // 延时变量

?= // 延时变量, 如果是第一次定义才起效, 如果在前面该变量已定义则忽略这句

+= // 附加, 它是即时变量还是延时变量取决于前面的定义

?=: // 如果这个变量在前面已经被定义,这句话就不会起效果。

实例:

A := $(C)
B = $(C)
C = abc

 

#D = 100ask

D ?= weidongshan


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

 

C += 123

执行:

make

 

结果:

A =
B = abc 123
D = weidongshan

分析:

1)
A := $(C)
:A为即时变量,在定义时即确定,由于刚开始C的值为空,所以A的值也为空。

2) B = (C):B为延时变量,只有使用到时它的值才确定,当执行make时,会解析Makefile里面的所用变量,所以先解析C=abc, 然后解析C+=123,此时C=abc123,

当执行:@echoB=(C):B为延时变量,只有使用到时它的值才确定,当执行make时,会解析Makefile里面的所用变量,所以先解析C=abc, 然后解析C+=123,此时,C=abc123,当执行:@echoB=(B) B的值为 abc 123。

 

3)
D ?= weidongshan
:D变量在前面没有定义,所以D的值为weidongshan,如果在前面添加D = 100ask,最后D的值为100ask。

我们还可以通过命令行存入变量的值
例如执行:

make D=123456

里面的D ?= weidongshan这句话便不再起作用。

结果:

A =
B = abc 123

D = 123456


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值