轻松玩转makefile | 变量与模式

前言

本文通过简单的几个示例,以及对同一个Makefile进行几个版本的迭代,帮助快速的理解变量和模式规则的使用。

1、回顾

在上一篇文章中,我们使用Makefile编译fun.c和main.c这两个文件,最终生成名为app的可执行文件。

fun.c的内容

#include <stdio.h>

void fun()
{
    printf("This is fun()!\n");
}

main.c的内容

extern void fun(); 

int main()
{ 
    fun(); 
    return 0; 
}

第一版Makefile

.PHONY:clean

all: main.o fun.o
    gcc -o app main.o fun.o
main.o : main.c 
    gcc -o main.o -c main.c
fun.o : fun.c 
    gcc -o fun.o -c fun.c
clean:
    rm app main.o fun.o

可以发现,Makefile里面有很多重复的内容,我们可以利用变量和模式规则对其进行优化。

2、自动变量

@:用于表示一个规则中的目标。当有多个目标时,@指的是其中任何导致规则命令被运行的自标。

$^:表示的是规则中的所有先决条件。

$<:表示的是规则中的第一个先决条件。

是不是看得有点晕?没关系,我们基于上面的Makefile,做一点点修改,把这些东西都打印出来看一下,就很清楚了。

.PHONY:clean

all: main.o fun.o
    gcc -o app main.o fun.o
    
    @echo "\$$@ = $@"
    @echo "$$^ = $^"
    @echo "$$< = $<"
    
main.o : main.c 
    gcc -o main.o -c main.c
fun.o : fun.c 
    gcc -o fun.o -c fun.c
clean:
    rm app main.o fun.o

运行make,终端打印如下内容

gcc -o app main.o fun.o
$@ = all
$^ = main.o fun.o
$< = main.o

了解到这些之后,我们再次修改

第二版Makefile

.PHONY:clean

all: main.o fun.o
    gcc -o app $^
    
main.o : main.c 
    gcc -o $@ -c $^
fun.o : fun.c 
    gcc -o $@ -c $^
clean:
    rm app main.o fun.o

3、变量的类别与赋值

变量的类别有<font color='red'>递归扩展变量</font>和<font color='green'>简单扩展变量</font>

3.1 递归扩展变量

这种只用一个“=”符号定义的变量被称为递归扩展变量

.PHONY:all

goal = $(mid)
mid = $(fun)
fun = test

all:
    @echo "goal = $(goal)"

运行make命令 ,打印如下

goal = test

3.2 简单扩展变量

用“:=”操作符来定义的,make只对其进行一次展开。

.PHONY:all

goal_A = hello 
mid_A  = $(goal_A) world
goal_A = test

goal_B := hello
mid_B  := $(goal_B) world
goal_B := test

all:
    @echo "mid_A = $(mid_A),mid_B= $(mid_B)"

运行make命令 ,打印如下

mid_A = test world,mid_B= hello world

3.3 变量条件赋值

用“?=”操作符来定义,如果变量没有被定义,将右边的值赋值给它,如果变量已经定义了,则不改变其原值。

.PHONY:all

funA = original
funA ?= replacement

funB ?= replacement
all:
    @echo "funA = $(funA),funB = $(funB)"

运行make命令 ,打印如下

funA = original,funB = replacement

3.4 变量追加赋值

通过“+=”实现追加赋值

.PHONY:all

objects = main.o fun.o
objects += append.o

all:
    @echo "objects = $(objects)"

运行make命令 ,打印如下

objects = main.o fun.o append.o

3.5 高级变量引用功能

在赋值的同时,完成文件名后缀替换操作

.PHONY:all

src = a.c b.c c.c
objs := $(src:.c=.o)

all:
    @echo "objs = $(objs)"

运行make命令 ,打印如下

objs = a.o b.o c.o

注意,src:.c=.o 冒号后面不能有空格,如果加了空格变成src: .c=.o ,运行make,打印的结果如下

objs = a.c b.c c.c

4、模式规则

一个模式规则的格式为:

%.o : %.c 
    command...

我们利用模式规则对第二版的Makefile进行优化

第三版Makefile


#替换掉这部分
#main.o : main.c 
#   gcc -o $@ -c $^
#fun.o : fun.c 
#   gcc -o $@ -c $^

.PHONY:clean

all: main.o fun.o
    gcc -o app $^
%.o : %.c
    gcc -o $@ -c $^
clean:
    rm app main.o fun.o

这里将两条构建目标文件的规则变成了一条

" % " 类似于通配符,%.c匹配所有以".c "结尾的文件,采用了模式以后,不管有多少个源文件,都可以用同一条规则,可以极大的简化Makefile

5、利用变量和模式规则优化Makefile

我们再对第三版的Makefile进行优化,将编译器,目标等都用变量替代,这样以后修改只需要改动变量部分就好了

第四版Makefile

.PHONY:clean

CC = gcc
RM = rm

TARGET = app
OBJS = main.o fun.o

$(TARGET) : $(OBJS)
    $(CC) -o $@ $^
%.o : %.c
    $(CC) -o $@ -c $^
clean:
    $(RM) $(TARGET) $(OBJS)

到这里,Makefile已经得到了很大的改善,但是我们可以看到OBJS = xxxx 这里,如果文件数量多,得一个个书写,还是不够智能。

下一篇文章,将介绍Makefile函数的使用,利用函数可以轻松管理好源文件和目标文件。
码字不易,点个赞再走吧!

欢迎关注我的同名公众号,这里有更多好料等着你哦!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,188评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,464评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,562评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,893评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,917评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,708评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,430评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,342评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,801评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,976评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,115评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,804评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,458评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,008评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,135评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,365评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,055评论 2 355

推荐阅读更多精彩内容

  • 什么是 Makefile 呢? Makefile 可以简单的认为是一个工程文件的编译规则,描述了整个工程的编译和链...
    JackHCC阅读 10,601评论 0 2
  • @(linux 编程)[开发技能, 工具使用] What is GNU Make Make 是控制工程中通过源码生...
    orientlu阅读 11,346评论 0 26
  • 前言 本文通过几个简单的示例,可以快速了解Makefile的基本使用方法,适用于编译我们平时练习所编写的小量代码。...
    知微之见阅读 385评论 0 0
  • Makefile的规则 在讲述这个Makefile之前,还是让我们先来粗略地看一看Makefile的规则。 tar...
    小y同学hh阅读 538评论 0 0
  • 1.Makefile规范 target 这 一 个 或 多 个 的 目 标 文 件 依 赖 于prerequisi...
    G风阅读 1,893评论 0 3