Makefile文件編寫

語法

target ... : prerequisites ...
    command
    ...
    ...
  • target可以是一個(gè)object file(目標(biāo)文件),也可以是一個(gè)執(zhí)行文件,還可以是一個(gè)標(biāo)簽(label)。對(duì)于標(biāo)簽這種特性,在后續(xù)的“偽目標(biāo)”章節(jié)中會(huì)有敘述。
  • prerequisites就是,要生成那個(gè)target所需要的文件或是目標(biāo)。
  • command也就是make需要執(zhí)行的命令。(任意的shell命令)

make工作

  • 默認(rèn)執(zhí)行 make 命令時(shí), GNU make在當(dāng)前目錄下依次搜索下面3個(gè)文件 "GNUmakefile", "makefile", "Makefile",
  • 如果找到,它會(huì)找文件中的第一個(gè)目標(biāo)文件(target),并把這個(gè)文件作為最終的目標(biāo)文件。
  • 如果target文件不存在,或是target所依賴的后面的 .o 文件的文件修改時(shí)間要比target這個(gè)文件新,那么,他就會(huì)執(zhí)行后面所定義的命令來生成target這個(gè)文件。
  • 如果target所依賴的文件也不存在,那么make會(huì)在當(dāng)前文件中找依賴文件,如果找到則再根據(jù)那一個(gè)規(guī)則生成依賴文件。(這有點(diǎn)像一個(gè)堆棧的過程)
  • 最終生成target文件

make 參數(shù)介紹

make 的參數(shù)有很多, 可以通過 make -h 去查看, 下面只介紹幾個(gè)我認(rèn)為比較有用的。

參數(shù) 含義
--debug[=<options>] 輸出make的調(diào)試信息, options 可以是 a, b, v
-j --jobs 同時(shí)運(yùn)行的命令的個(gè)數(shù), 也就是多線程執(zhí)行 Makefile
-r --no-builtin-rules 禁止使用任何隱含規(guī)則
-R --no-builtin-variabes 禁止使用任何作用于變量上的隱含規(guī)則
-B --always-make 假設(shè)所有目標(biāo)都有更新, 即強(qiáng)制重編譯

注意

  • 所有的命令前要用tab分割

變量

定義變量(= or := )
  • := 只能使用前面定義好的變量
  • = 可以使用后面定義的變量
OBJS = programA.o programB.o
OBJS-ADD = $(OBJS) programC.o
# 或者
OBJS := programA.o programB.o
OBJS-ADD := $(OBJS) programC.o
使用變量 $()
變量追加值 +=
SRCS := programA.c programB.c programC.c
SRCS += programD.c
變量覆蓋 override
目標(biāo)變量

作用是使變量的作用域僅限于這個(gè)目標(biāo)(target), 而不像之前例子中定義的變量, 對(duì)整個(gè)Makefile都有效.

語法:

  • <target ...> :: <variable-assignment>
  • <target ...> :: override <variable-assignment> (override作用參見 變量覆蓋的介紹)
# Makefile 內(nèi)容
SRCS := programA.c programB.c programC.c

target1: TARGET1-SRCS := programD.c
target1:
    @echo "SRCS: " $(SRCS)
    @echo "SRCS: " $(TARGET1-SRCS)

target2:
    @echo "SRCS: " $(SRCS)
    @echo "SRCS: " $(TARGET1-SRCS)

# bash中執(zhí)行make
$ make target1
SRCS:  programA.c programB.c programC.c
SRCS:  programD.c

$ make target2     <-- target2中顯示不了 $(TARGET1-SRCS)
SRCS:  programA.c programB.c programC.c
SRCS:

Makefile 命令前綴(@ or -)

Makefile 中書寫shell命令時(shí)可以加2種前綴 @ 和 -, 或者不用前綴.

  • 不用前綴 。輸出執(zhí)行的命令以及命令執(zhí)行的結(jié)果, 出錯(cuò)的話停止執(zhí)行
  • 前綴 @ 只輸出命令執(zhí)行的結(jié)果, 出錯(cuò)的話停止執(zhí)行
  • 前綴 - 命令執(zhí)行有錯(cuò)的話, 忽略錯(cuò)誤, 繼續(xù)執(zhí)行

偽目標(biāo)

偽目標(biāo)并不是一個(gè)"目標(biāo)(target)", 不像真正的目標(biāo)那樣會(huì)生成一個(gè)目標(biāo)文件.

典型的偽目標(biāo)是 Makefile 中用來清理編譯過程中中間文件的 clean 偽目標(biāo), 一般格式如下:

.PHONY: clean   <-- 這句沒有也行, 但是最好加上
clean:
    -rm -f *.o      

引用其他的 Makefile

語法: include <filename> (filename 可以包含通配符和路徑)

# Makefile 內(nèi)容
all:
    @echo "主 Makefile begin"
    @make other-all
    @echo "主 Makefile end"

include ./other/Makefile

# ./other/Makefile 內(nèi)容
other-all:
    @echo "other makefile begin"
    @echo "other makefile end"

# bash中執(zhí)行 make
$ ll
total 20K
-rw-r--r-- 1 wangyubin wangyubin  125 Sep 23 16:13 Makefile
-rw-r--r-- 1 wangyubin wangyubin  11K Sep 23 16:15 makefile.org   <-- 這個(gè)文件不用管
drwxr-xr-x 2 wangyubin wangyubin 4.0K Sep 23 16:11 other
$ ll other/
total 4.0K
-rw-r--r-- 1 wangyubin wangyubin 71 Sep 23 16:11 Makefile

$ make
主 Makefile begin
make[1]: Entering directory `/path/to/test/makefile'
other makefile begin
other makefile end
make[1]: Leaving directory `/path/to/test/makefile'
主 Makefile end

Makefile 隱含規(guī)則

這里只列一個(gè)和編譯C相關(guān)的.

編譯C時(shí),<n>.o 的目標(biāo)會(huì)自動(dòng)推導(dǎo)為 <n>.c

# Makefile 中
main : main.o
    gcc -o main main.o

#會(huì)自動(dòng)變?yōu)?
main : main.o
    gcc -o main main.o

main.o: main.c    <-- main.o 這個(gè)目標(biāo)是隱含生成的
    gcc -c main.c

自動(dòng)變量

自動(dòng)變量 含義
$@ 目標(biāo)集合
$% 當(dāng)目標(biāo)是函數(shù)庫文件時(shí), 表示其中的目標(biāo)文件名
$< 第一個(gè)依賴目標(biāo). 如果依賴目標(biāo)是多個(gè), 逐個(gè)表示依賴目標(biāo)
$? 比目標(biāo)新的依賴目標(biāo)的集合
$^ 所有依賴目標(biāo)的集合, 會(huì)去除重復(fù)的依賴目標(biāo)
$+ 所有依賴目標(biāo)的集合, 不會(huì)去除重復(fù)的依賴目標(biāo)
$* 這個(gè)是GNU make特有的, 其它的make不一定支持
all

一種簡寫,可以讓多個(gè)目標(biāo)操作順次執(zhí)行

all: server.out client.out
objects = server.cpp
server.out : $(objects)
    g++ -o server.out $(objects)

client.out : client.cpp
    g++ -o client.out client.cpp

直接 make 或 make all 的話會(huì)執(zhí)行 server.out 和client.out 的編譯命令,后面不加參數(shù)的話,會(huì)把第一個(gè)目標(biāo)作為默認(rèn)的。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,983評(píng)論 6 537
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,772評(píng)論 3 422
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,947評(píng)論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,201評(píng)論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,960評(píng)論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,350評(píng)論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,406評(píng)論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,549評(píng)論 0 289
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,104評(píng)論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,914評(píng)論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,089評(píng)論 1 371
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,647評(píng)論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,340評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,753評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,007評(píng)論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,834評(píng)論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,106評(píng)論 2 375

推薦閱讀更多精彩內(nèi)容

  • 來自陳浩的一片老文,但絕對(duì)營養(yǎng)。 示例工程:3 個(gè)頭文件*.h,和 8 個(gè) C 文件*.c。 初 編譯過程,源文件...
    周筱魯閱讀 4,718評(píng)論 0 17
  • 1 簡介 Makefile是用來完成構(gòu)建,編譯整個(gè)工程文件的工具。所以,Makefile定義了一系列的規(guī)則,按照這...
    wit_yuan閱讀 972評(píng)論 0 1
  • makefile關(guān)系到整個(gè)工程的編譯規(guī)則,一個(gè)工程中的源文件不計(jì)其數(shù),按其類型、功能、模塊分別放在若干的目錄當(dāng)中,...
    Joe_HUST閱讀 1,890評(píng)論 0 3
  • @(linux 編程)[開發(fā)技能, 工具使用] What is GNU Make Make 是控制工程中通過源碼生...
    orientlu閱讀 11,365評(píng)論 0 26
  • 2017年10月27日,星期五,晴。 今天早晨,收到家長發(fā)來的信息。打開一看,是一位家長說她孩子早晨帶手機(jī)到校了,...
    夏日梅語閱讀 449評(píng)論 0 1