makefile 簡介

1.簡單的命令格式
target : prerequisites
recipe

target 通常是一個程序生出的文件名,比如對象文件(.o),也可能是一個將要執行的動作命令 比如 clean
prerequisites 通常是一個或多個生成 target 所需要的文件
recipe 是 make 要執行的方法,可以一個或多個,可以使一行或多行,注意每行都需要 tab 縮進,.RECIPEPREFIX變量可以修改 tab 爲其他字符

在一條命令中,recipe 和 prerequisites 都是爲如何生成 target 服務的
但是,執行 recipe 不一定要有 prerequisites,比如 clean
有一些 target ,不是file,不作爲任何 rules 的 prerequisites,稱爲 空頭target (phony target),執行時,必須顯示指明 target ,如 make clean
e.g.
edit : main.o kbd.o command.o display.o insert.o search.o files.o utils.o
	cc -o edit main.o kbd.o command.o display.o insert.o search.o files.o utils.o
main.o : main.c defs.h
	cc -c main.c
kbd.o : kbd.c defs.h command.h
	cc -c kbd.c
command.o : command.c defs.h command.h
	cc -c command.c
display.o : display.c defs.h buffer.h
	cc -c display.c
insert.o : insert.c defs.h buffer.h
	cc -c insert.c
search.o : search.c defs.h buffer.h
	cc -c search.c
files.o : files.c defs.h buffer.h command.h
	cc -c files.c
utils.o : utils.c defs.h
	cc -c utils.c
clean :
	rm edit main.o kbd.o command.o display.o insert.o search.o files.o utils.o
2.make 處理 makefile 
默認 make 首先處理第一個不以.開頭的 target ,在執行其 recipe 前,先檢查 prerequisites 和他們各自的 prerequisites。 只要 prerequisites 比 target 新或者 target 不存在,其 recipe 就會執行
上面例子中,首先處理的 target 是 edit,檢查 prerequisites 中所有 .o 文件以及對應的 prerequisites.
比如改變了 command.h,則重新編譯kbd.o, command.o, files.o 三個文件,最後重新鏈接所有 .o 文件

3.使用變量
上列中重複使用了 3個文件列表,如果我們添加或刪除了某個 target 或 prerequisites,不得不同時修改3處,容易發生遺漏
這時可以使用變量簡化程序
objects = main.o kbd.o command.o display.o insert.o search.o files.o utils.o
edit : $(objects)
	cc -o edit $(objects)
...
clean :
	rm edit $(objects)

4.使用隱式規則推斷 recipe
我們總是用 cc -c xx.c xx.h -o xx.o 命令編譯 .o 文件,這在 make 中是一條隱式規則,我們可以省略 prerequisites 中的與 .o 文件對應的 .c 文件以及 recipe
上面的列子簡化爲

objects = main.o kbd.o command.o display.o insert.o search.o files.o utils.o
edit : $(objects)
	cc -o edit $(objects)
main.o : defs.h
kbd.o : defs.h command.h
command.o : defs.h command.h
display.o : defs.h buffer.h
insert.o : defs.h buffer.h
search.o : defs.h buffer.h
files.o : defs.h buffer.h command.h
utils.o : defs.h

.PHONY : clean
clean :
	rm edit $(objects)<span style="font-family: Arial, Helvetica, sans-serif; background-color: rgb(255, 255, 255);">	</span>
5.如果 makefile 只使用了隱式規則,我們還可以通過分組入口(group entries)編寫
objects = main.o kbd.o command.o display.o insert.o search.o files.o utils.o
edit : $(objects)
	cc -o edit $(objects)
$(objects) : defs.h
kbd.o command.o files.o : command.h
insert.o display.o files.o search.o : buffer.h
6.除了編譯文件,makefile 還可以用來刪除文件和可執行文件清理目錄
.PHONY : clean
clean :
rm edit $(objects)

.PHONY : clean 避免 target clean 和 名爲 clean 的文件混淆,clean 不應該放在 makefile的開始位置,否則他會成爲默認 target, 而且它不是任何 target 的 prerequisites ,只能通過 make clean 執行

參考:《GNU make manual》2.An Introduction to Makefiles

手冊下載地址:http://www.gnu.org/software/make/manual/

發佈了41 篇原創文章 · 獲贊 1 · 訪問量 1萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章