Makefile 变量

跟 C 语言一样 Makefile 也支持变量的,先看一下前面的例子:

 main: main.o input.o calcu.o

         gcc -o main main.o input.o calcu.o

上述 Makefile 语句中,main.o input.o 和 calcue.o 这三个依赖文件,我们输入了两遍,我们

这个 Makefile 比较小,如果 Makefile 复杂的时候这种重复输入的工作就会非常费时间,而且非

常容易输错,为了解决这个问题,Makefile 加入了变量支持。不像 C 语言中的变量有 int、char

等各种类型,Makefile 中的变量都是字符串!类似 C 语言中的宏。使用变量将上面的代码修改,

修改以后如下所示:

示例代码 3.4.2.1 Makefile 变量使用

1 #Makefile 变量的使用

2 objects = main.o input.o calcu.o

3 main: $(objects)

4 gcc -o main $(objects)

  • 注释开头要 用符号“#”,
  • Makefile 中变量的引用方法是“$(变量名)”

中我们在定义变量 objects 的时候使用“=”对其进行了赋值,Makefile

变量的赋值符还有其它两个“:=”和“?=”,我们来看一下这三种赋值符的区别:

  • 使用“=”在给变量的赋值的时候,是变量的真实值取决于它所引用的变量的最后一次有效值。
  • 为赋值符“:=”不会使用后面定义的变量.
  • “?=”是一个很有用的赋值符,curname ?= zuozhongkai,,如果变量 curname 前面没有被赋值,那么此变量就是“zuozhongkai”, 如果前面已经赋过值了,那么就使用前面赋的值。

变量追加“+=”,有时候我们需要给前面已经定义好的变量添加一些字符串进

去,此时就要使用到符号“+=”。

将对应的.c 源文件编译为.o 文件,每一个 C 文件都要写一个对

应的规则,如果工程中 C 文件很多的话显然不能这么做。

通过模式规则我们就可以使用一条规则来将所有的.c 文件编译为对应的.o 文件。

%.o : %.c         命令

“%”表示长度任意的非空字符串,比如“%.c”就是所有的以.c 结尾的

文件,类似与通配符,a.%.c 就表示以 a.开头,以.c 结束的所有文件。

C语言工程一个通用的makefile文件看我这篇文章C语言工程的通用makefile模板(干货!!!)

Makefile 自动化变量

每一次对模式规则进行解析的时候

都会是不同的目标和依赖文件,而命令只有一行,如何通过一行命令来从不同的依赖文件中生

成对应的目标?自动化变量就是完成这个功能的!

makefile基础知识学习篇2_makefile基础知识

常用的三种:$@(目标集合)、$<和$^(所有依赖文件集合,空格隔开)

1 objects = main.o input.o calcu.o

2 main: $(objects)

3         gcc -o main $(objects)

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

8 clean:

9         rm *.o

10       rm main

Makefile 伪目标,看我这篇文章makefile伪目标详解

Makefile 条件判断

语法有两种如下:

<条件关键字>

        <条件为真时执行的语句>

endif

以及:

<条件关键字>

        <条件为真时执行的语句>

else

        <条件为假时执行的语句>

endif

其中条件关键字有 4 个:ifeq、ifneq、ifdef 和 ifndef,这四个关键字其实分为两对、ifeq 与

ifneq、ifdef 与 ifndef,先来看一下 ifeq 和 ifneq,ifeq 用来判断是否相等,ifneq 就是判断是否不

相等,ifeq 用法如下:

ifeq (<参数 1>, <参数 2>)

ifeq ‘<参数 1 >’,‘ <参数 2>’

ifeq “<参数 1>”, “<参数 2>”

ifeq “<参数 1>”, ‘<参数 2>’

ifeq ‘<参数 1>’, “<参数 2>”

上述用法中都是用来比较“参数 1”和“参数 2”是否相同,如果相同则为真,“参数 1”和

“参数 2”可以为函数返回值。ifneq 的用法类似,只不过 ifneq 是用来了比较“参数 1”和“参

数 2”是否不相等,如果不相等的话就为真。

ifdef 和 ifndef 的用法如下:

ifdef <变量名>

如果“变量名”的值非空,那么表示表达式为真,否则表达式为假。“变量名”同样可以是

一个函数的返回值。ifndef 用法类似,但是含义用户 ifdef 相反。

makefile内置特殊变量“VPATH”,make会自动去找VPATH中的源文件。

如果没有指明这个变量,make只会在当前的目录中去找寻依赖文件和目标文件。如果定义了这个变量,那么,make就会在当当前目录找不到的情况下,到所指定的目录中去找寻文件了。

Makefile静态模式

静态模式: :

targets定义了一系列的目标文件,可以有通配符。是目标的一个集合。

target-parrtern是指明了targets的子集合,也就是真正的目标集。

prereq-parrterns是目标集的依赖模式,它对target-parrtern形成的模式再进行一次依赖目标的定义。 

$(OBJ):%.o:%.c

    $(CC) -c $(CFLAGS) $< -o $@

使用的就是静态模式,其中指明了我们的目标从$OBJ 中获取,“%.o”表明要所有以“.o”结尾的目标,也就是“main.o fun.o”,

也就是变量$OBJ  集合的模式,而依赖模式“%.c”则取模式“%.o”的“%”,也就是“main fun”,并为其加下“.c”的后缀,

于是,我们的依赖目标就是“main.c fun.c”。

Makefile 函数使用

Makefile 支持函数,类似 C 语言一样,Makefile 中的函数是已经定义好的,我们直接使用,

不支持我们自定义函数。make 所支持的函数不多,但是绝对够我们使用了,函数的用法如下:

$(函数名 参数集合)

或者:

${函数名 参数集合}

可以看出,调用函数和调用普通变量一样,使用符号“$”来标识。参数集合是函数的多个

参数,参数之间以逗号“,”隔开,函数名和参数之间以“空格”分隔开,函数的调用以“$”开

头。接下来我们介绍几个常用的函数,其它的函数大家可以参考《跟我一起写 Makefile》这份

文档。

1、函数 subst

函数 subst 用来完成字符串替换,调用形式如下:

$(subst ,,)

此函数的功能是将字符串中的内容替换为,函数返回被替换以后的字符

串,比如如下示例:

$(subst zzk,ZZK,my name is zzk)

把字符串“my name is zzk”中的“zzk”替换为“ZZK”,替换完成以后的字符串为“my name 

is ZZK”。

2、函数 patsubst

函数 patsubst 用来完成模式字符串替换,使用方法如下:

$(patsubst ,,)

此函数查找字符串(如果字符串中有空格,则认为是另外一个字符串)中的单词是否符合模式,如果匹配就用来

替换掉,可以使用通配符“%”,表示任意长度的字符串,函数返回值就是替换后的字

符串。如果中也包涵“%”,那么中的“%”将是中的那个

“%”所代表的字符串,比如:

$(patsubst %.c,%.o,a.c b.c c.c)

将字符串“a.c b.c c.c”中的所有符合“%.c”的字符串,替换为“%.o”,替换完成以后的字

符串为“a.o b.o c.o”。字符串是有空格分隔的

3、函数 dir

函数 dir 用来获取目录,使用方法如下:

$(dir )

此函数用来从文件名序列中提取出目录部分,返回值是文件名序列的目录

部分,比如:

$(dir /a.c>)

提取文件“/src/a.c”的目录部分,也就是“/src”。

4、函数 notdir

函数 notdir 看名字就是知道去除文件中的目录部分,也就是提取文件名,用法如下:

$(notdir )

此函数用与从文件名序列中提取出文件名非目录部分,比如:

$(notdir /a.c>)

提取文件“/src/a.c”中的非目录部分,也就是文件名“a.c”。

5、函数 foreach

foreach 函数用来完成循环,用法如下:

$(foreach , ,)

此函数的意思就是把参数中的单词逐一取出来放到参数中,然后再执行所

包含的表达式。每次都会返回一个字符串,循环的过程中,中所包含的每个字符串

会以空格隔开,最后当整个循环结束时,所返回的每个字符串所组成的整个字符串将会是

函数 foreach 函数的返回值。

6、函数 wildcard

通配符“%”只能用在规则中,只有在规则中它才会展开,如果在变量定义和函数使用时,

通配符不会自动展开,这个时候就要用到函数 wildcard,使用方法如下:

$(wildcard PATTERN…)

比如:

$(wildcard *.c)

上面的代码是用来获取当前目录下所有的.c 文件,类似“%”。