关于Makefile

本文参考:
Make 命令教程 - 阮一峰

本学期正在上一门具有“USC神课”之美誉的CSCI402 - Operating Systems。这门课非常的严谨,编程作业的spec写得相当之详细,要求也相当地规范。编程作业要求使用makefile来编译所有的程序,因此该篇文章用于记录我从makefile小白到入门的过程。

1. 什么是Make,什么是Makefile?

In software development, Make is a build automation tool that automatically builds executable programs and libraries from source code by reading files called Makefiles which specify how to derive the target program.
—— wiki

从wiki的解释可以知道,Make是一种自动化的构建工具。那么什么是构建(build)呢?这又先涉及到什么是编译(compile)。编译是指将源代码转换为计算机可执行代码的过程。而关于编译的安排,即先编译哪一部分,后编译哪一部分,则叫做构建(build)。

Make通过读取叫做Makefile的文件,来按照开发者的安排对源代码进行构建。

2. 如何使用make?

make的意思即为“制作”,关于制作一样东西,需要指明的是:

  • 要制作什么?
  • 它依赖于什么?
  • 该怎么通过它的依赖来构建它?
  • 当它的依赖有变动时,该怎么处理它们?

比如,在一个名叫rules.txt文件中,有如下规定:

a.txt: b.txt c.txt
    cat b.txt c.txt > a.txt

这个文件所要表达的意思是:

  • Make需要构建名叫a.txt的文件。
  • 它依赖于b.txt与c.txt文件的存在。
  • 通过cat命令将b.txt与c.txt合并的方式制作a.txt

需要再命令行执行:

$ make --file=rules.txt

意思是根据rules.txt文件中的规定来构建。

3. 如何编写Makefile

3.1 基本格式

Makefile文件由一系列规则(rules)构成。每条规则的形式如下。

<target> : <prerequisites> 
[tab]  <commands>

上面第一行冒号前面的部分,叫做"目标"(target),冒号后面的部分叫做"前置条件"(prerequisites);第二行必须由一个tab键起首,后面跟着"命令"(commands)。

"目标"是必需的,不可省略;"前置条件"和"命令"都是可选的,但是两者之中必须至少存在一个

3.2 目标(target)

一个目标引领一条规则。目标即是需要构建的对象,可以是文件名,如上所示;也可以是某个操作,即伪目标(phony target)。

例如:

clean:
      rm *.o

意思为构建一个clean操作,该操作用于移除所有.o文件。

执行的操作是:

make clean

效果是当前目录会生成一个clean操作,并移除所有.o文件。

但是,如果当前目录中,正好有一个文件叫做clean,那么这个命令不会执行。因为Make发现clean文件已经存在,就认为没有必要重新构建了,就不会执行指定的rm命令。

为了避免这种情况,可以明确声明clean是"伪目标",写法如下。

.PHONY: clean
clean:
        rm *.o temp

效果是make就不会去检查是否存在一个叫做clean的文件,而是每次运行都执行对应的命令。

如果Make命令运行时没有指定目标,默认会执行Makefile文件的第一个目标

make

3.3 前置条件(prerequisite)

前置条件通常是一组文件名,之间用空格分隔。它指定了"目标"是否需要重新构建的判断标准:

  • 目标不存在,需要构建
  • 只要有一个前置文件不存在,或者有过更新(前置文件的last-modification时间戳比目标的时间戳新),"目标"就需要重新构建。
  • 其他情况不会重新构建

当某个前置条件中的文件不存在时,需要再写一条规则,来生成所需要的文件。

比如:

result.txt: source.txt
    cp source.txt result.txt
source.txt:
    echo "this is the source" > source.txt

构建result.txt需要source.txt,但是source.txt可能不存在。如果不存在,或者有过更新,则按照第二条规则来构建出source.txt。

如果执行如下命令:

$ make result.txt
$ make result.txt

第一次执行会先新建 source.txt,然后再新建 result.txt。第二次执行,Make发现 source.txt 没有变动(时间戳晚于 result.txt),就不会执行任何操作,result.txt 也不会重新生成。

如果需要生成多个文件,可以巧妙利用伪目标来实现:

source: file1 file2 file3

在命令行中:

$ make source

3.4 命令(commands)

命令(commands)表示如何构建(更新)目标文件,由一行或多行的Shell命令组成。

每行命令之前必须有一个tab键。需要注意的是,每行命令在一个单独的shell中执行。这些Shell之间没有继承关系。

var-lost:
    export foo=bar
    echo "foo=[$$foo]"

上面代码执行后make var-lost,取不到foo的值。因为两行命令在两个不同的进程执行。一个解决办法是将两行命令写在一行,中间用分号分隔。

有三个解决办法:

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

推荐阅读更多精彩内容