語法
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)的。