注:這里的makefile指的是gnu makefile,可能跟其他的makefile略有不同。若有不同,我盡量提及。而且這也不算是一篇教程,只是我在用make的時候記得一些筆記。推薦看<
>。另外網上有一篇《跟我一起寫makefile》也很好。
makefile規則分為三部分:目標(target)、條件(prerequisite)、命令(commands)。
target:prereq1...prereqn
commans
1.自動變量
$@:代表目標文件名
$%:The filename element of an archive member specification.
$<:條件列表中的第一個條件的文件名
$?:條件列表中所有比目標新的那些條件的文件名,以空格分割
$^:條件列表中所有條件的文件名,以空格分割。如果列表中有重復的條件,則會被刪掉。
$+:跟$^類似,區別在不會去掉重復的條件。
$*:目標文件名的詞干部分(去掉擴展名剩下的部分)
自動變量只能用在規則的命令部分,因為這些變量是make匹配到規則的目標和條件后才設置值的。
舉個例子說明一下。假設某個工程有三個文件:cal.cpp,cal.h,test.cpp(下面再有說明時也以此為例)。makefile如下:
OBJS=test.o cal.o cal.o
TARGET=cal.exe
all:$(TARGET)
$(TARGET):$(OBJS)
@echo $@
@echo $<
@echo $^
@echo $?
@echo $+
@echo $%
@echo $*
輸出為:
cal.exe
test.o
test.o cal.o
test.o cal.o
test.o cal.o cal.o
ECHO is off.
ECHO is off.
至于最后兩個為什么會這樣輸出我還不太清楚,有清楚地麻煩告訴我一下這兩個怎么用。先謝謝了。
2.模式規則
模式規則也是規則,也要滿足make的規則形式,分為目標、條件、命令三個部分。只是目標、條件中的文件名的詞干部分用%代替了,這個%跟shell中的*類似,代表任意長度的字符串。還以上面說得工程為例,makefile如下:
CPPFLAGS= -g
CC=g++
test:cal.o test.o
cal.o:cal.h
make的時候,輸出如下內容:
g++ -g -c -o test.o test.cpp
g++ -g -c -o cal.o cal.cpp
g++ test.o cal.o -o test
這個makefile之所以能夠正確被處理,是因為make內建了一些規則。例如:
%.o: %.c
$(COMPILE.c) $(OUTPUT_OPTION) $<
%.o: %.cpp
$(COMPILE.cpp) $(OUTPUT_OPTION) $<
以及
%: %.o
$(LINK.o) $^ $(LOADLIBES) $(LDLIBS) -o $@
這些規則里面的$變量都是make的標準變量(還有上面的CPPFLAGS,CC也是。另外還有一些別的),這些標準變量有的有默認值(例如CC的默認值是cc),有的沒有。make在處理makefile文件時,如果沒有顯式的規則,那么就會查找是否有隱式的可用規則,如果找到就會利用隱式規則來生成目標。
2.1靜態模式規則
靜態模式規則指的是類似下面的規則:
$(OBJECTS): %.o: %c
$(CC) -c $(CFLAGS) $< -o $@
這跟上面的模式規則類似,不同在于將規則的適用范圍限制在了$(OBJECTS)。也就是只針對這些目標來應用規則。
2.2后綴規則
后綴規則是指用一個或者兩個后綴連接起來作為目標,同時省略掉條件部分。如下:
.c.o:
$(COMPILE.c) $(OUTPUT_OPTION) $<
容易混淆的地方是條件后綴在前,目標后綴在后。
這跟上面的規則:
%.o: %.c
$(COMPILE.c) $(OUTPUT_OPTION) $<
功能是一樣的,只是為了兼容一些老的make系統。
需要注意的是,這里用到的后綴必須在已知后綴列表里面。有個專用的目標(target)——.SUFFIXES——用來設定后綴列表。例如:.SUFFIXES:.pdf .o .c 。而.SUFFIXES:(也就是后面列表為空)用來清空后綴列表。
3.自動依賴生成
還沒發現這個在實際應用中有多方便,有興趣的可以看看我開始推薦的兩個文檔。