二、Makefile文件的格式

2018-02-24 16:09 更新

構(gòu)建規(guī)則都寫在Makefile文件里面,要學(xué)會(huì)如何Make命令,就必須學(xué)會(huì)如何編寫Makefile文件。

2.1 概述

Makefile文件由一系列規(guī)則(rules)構(gòu)成。每條規(guī)則的形式如下。

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

上面第一行冒號(hào)前面的部分,叫做"目標(biāo)"(target),冒號(hào)后面的部分叫做"前置條件"(prerequisites);第二行必須由一個(gè)tab鍵起首,后面跟著"命令"(commands)。

"目標(biāo)"是必需的,不可省略;"前置條件"和"命令"都是可選的,但是兩者之中必須至少存在一個(gè)。

每條規(guī)則就明確兩件事:構(gòu)建目標(biāo)的前置條件是什么,以及如何構(gòu)建。下面就詳細(xì)講解,每條規(guī)則的這三個(gè)組成部分。

2.2 目標(biāo)(target)

一個(gè)目標(biāo)(target)就構(gòu)成一條規(guī)則。目標(biāo)通常是文件名,指明Make命令所要構(gòu)建的對(duì)象,比如上文的 a.txt 。目標(biāo)可以是一個(gè)文件名,也可以是多個(gè)文件名,之間用空格分隔。

除了文件名,目標(biāo)還可以是某個(gè)操作的名字,這稱為"偽目標(biāo)"(phony target)。

clean:
      rm *.o

上面代碼的目標(biāo)是clean,它不是文件名,而是一個(gè)操作的名字,屬于"偽目標(biāo) ",作用是刪除對(duì)象文件。

$ make  clean

但是,如果當(dāng)前目錄中,正好有一個(gè)文件叫做clean,那么這個(gè)命令不會(huì)執(zhí)行。因?yàn)镸ake發(fā)現(xiàn)clean文件已經(jīng)存在,就認(rèn)為沒有必要重新構(gòu)建了,就不會(huì)執(zhí)行指定的rm命令。

為了避免這種情況,可以明確聲明clean是"偽目標(biāo)",寫法如下。

.PHONY: clean
clean:
        rm *.o temp

聲明clean是"偽目標(biāo)"之后,make就不會(huì)去檢查是否存在一個(gè)叫做clean的文件,而是每次運(yùn)行都執(zhí)行對(duì)應(yīng)的命令。像.PHONY這樣的內(nèi)置目標(biāo)名還有不少,可以查看手冊(cè)。

如果Make命令運(yùn)行時(shí)沒有指定目標(biāo),默認(rèn)會(huì)執(zhí)行Makefile文件的第一個(gè)目標(biāo)。

$ make

上面代碼執(zhí)行Makefile文件的第一個(gè)目標(biāo)。

2.3 前置條件(prerequisites)

前置條件通常是一組文件名,之間用空格分隔。它指定了"目標(biāo)"是否重新構(gòu)建的判斷標(biāo)準(zhǔn):只要有一個(gè)前置文件不存在,或者有過更新(前置文件的last-modification時(shí)間戳比目標(biāo)的時(shí)間戳新),"目標(biāo)"就需要重新構(gòu)建。

result.txt: source.txt
    cp source.txt result.txt

上面代碼中,構(gòu)建 result.txt 的前置條件是 source.txt 。如果當(dāng)前目錄中,source.txt 已經(jīng)存在,那么make result.txt可以正常運(yùn)行,否則必須再寫一條規(guī)則,來生成 source.txt 。

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

上面代碼中,source.txt后面沒有前置條件,就意味著它跟其他文件都無關(guān),只要這個(gè)文件還不存在,每次調(diào)用make source.txt,它都會(huì)生成。

$ make result.txt
$ make result.txt

上面命令連續(xù)執(zhí)行兩次make result.txt。第一次執(zhí)行會(huì)先新建 source.txt,然后再新建 result.txt。第二次執(zhí)行,Make發(fā)現(xiàn) source.txt 沒有變動(dòng)(時(shí)間戳晚于 result.txt),就不會(huì)執(zhí)行任何操作,result.txt 也不會(huì)重新生成。

如果需要生成多個(gè)文件,往往采用下面的寫法。

source: file1 file2 file3

上面代碼中,source 是一個(gè)偽目標(biāo),只有三個(gè)前置文件,沒有任何對(duì)應(yīng)的命令。

$ make source

執(zhí)行make source命令后,就會(huì)一次性生成 file1,file2,file3 三個(gè)文件。這比下面的寫法要方便很多。

$ make file1
$ make file2
$ make file3

2.4 命令(commands)

命令(commands)表示如何更新目標(biāo)文件,由一行或多行的Shell命令組成。它是構(gòu)建"目標(biāo)"的具體指令,它的運(yùn)行結(jié)果通常就是生成目標(biāo)文件。

每行命令之前必須有一個(gè)tab鍵。如果想用其他鍵,可以用內(nèi)置變量.RECIPEPREFIX聲明。

.RECIPEPREFIX = >
all:
> echo Hello, world

上面代碼用.RECIPEPREFIX指定,大于號(hào)(>)替代tab鍵。所以,每一行命令的起首變成了大于號(hào),而不是tab鍵。

需要注意的是,每行命令在一個(gè)單獨(dú)的shell中執(zhí)行。這些Shell之間沒有繼承關(guān)系。

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

上面代碼執(zhí)行后(make var-lost),取不到foo的值。因?yàn)閮尚忻钤趦蓚€(gè)不同的進(jìn)程執(zhí)行。一個(gè)解決辦法是將兩行命令寫在一行,中間用分號(hào)分隔。

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

另一個(gè)解決辦法是在換行符前加反斜杠轉(zhuǎn)義。

var-kept:
    export foo=bar; \
    echo "foo=[$$foo]"

最后一個(gè)方法是加上.ONESHELL:命令。

.ONESHELL:
var-kept:
    export foo=bar; 
    echo "foo=[$$foo]"
以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)