個人理解:shell
拿clean舉例,若是make完成後,本身另外定義一個名叫clean的文件,再執行make clean時,將不會執行rm命令。 命令行
爲了不出現這個問題,須要.PHONY: clean遞歸
=======================================================================================效率
所謂僞目標就是這樣一個目標,它不表明一個真正的文件名,在執行make時能夠指定這個目標來執行其所在規則定義的命令,有時咱們將一個僞目標成爲標籤。file
那麼到底什麼是僞目標呢?可能做爲初學者還不會在意這個問題,下面咱們來看下咱們將在何時須要它。循環
首先來看下面一個例子:程序
當前目錄下只有一個myls1.c,因而爲了讓程序讓makefile來管理,寫了一個以下的簡單的makefile。並行
執行:makefile
你們會發現,真的能夠利用這個makefile管理當前的工程,也能如期按照咱們的要求生成執行文件myls。di
執行make clean,這樣就能夠刪除可執行程序。
接着我作了個手腳,在當前目錄下創建一個叫clean的文件,那麼這樣執行的效果是如何?
那麼這個時候爲何又不能執行了?在個人makefile中其實並無修改任何東西,爲何這個時候已經能管理工程的makefile又不能來管理文件了。
那要解決這個問題就是添加兩行,修改後的makefile以下:
再次返回執行:
這樣就解決了問題,那具體的緣由是什麼?
在makefile中咱們使用僞目標就能夠解決上述的問題,那爲何要使用僞目標,一種就是如例題,爲了不在makefile中定義的只執行命令的目標和工做目錄下的實際文件出現名字衝突,另外一種是提交執行makefile時的效率。
第一種狀況:
若是咱們須要書寫這樣的一個規則:規則所定義的命令不是去建立目標文件,而是經過make命令行明確指定它來執行一些特色的命令,就像例題中的clean。當文件夾中沒有clean這個文件的時候,咱們輸入「make clean」能按照初衷執行,可是一旦文件夾中出現clean文件,咱們再次輸入「make clean」,因爲這個規則沒有任何依賴文件,因此目標被認爲是最新的而不去執行規則所定義的命令。因此rm命令不會被執行。爲了解決問題,咱們將目標clean定義成僞目標。
也就是添加:
.PHONY:clean
那麼目錄中不管是否有clean文件,只要輸入「make clean」就能執行rm命令了。
當一個目標被聲明爲僞目標後,make在執行規則時不會去試圖去查找隱含規則來建立它。這樣就提升了make的執行效率,也不用擔憂因爲目標和文件名重名了。
第二種狀況:
僞目標的另外一種使用場合時在make的並行和遞歸執行過程當中。
給了例子:
SUBDIRS=foo bar baz
Subdirs:
for dir in $(SUBDIRS)
do
$(MAKE) –C $$dir
done
若是這樣寫,會出現幾個問題:
一、 當子目錄執行make出現錯誤,make不會退出;
二、 使用這種shell的循環方式時,沒有用到make對目錄的並行處理功能。
有了僞目標就能夠解決上面的兩個問題。
SUBDIRS=foo bar baz
.PHONY:subdirs $(SUBDIRS)
subdirs: $(SUBDIRS)
$(SUBDIRS):
$(MAKE) –C $@
通常狀況下,一個僞目標不做爲另外一個目標的依賴。當一個僞目標沒有做爲任何目標的依賴時,咱們只能經過make命令來明確指定它爲make的終極目標,來執行它所在規則所定義的命令。
還有一個特別的僞目標——all,若是咱們在一個目錄下建立多個可執行程序,咱們能夠將全部程序的重建規則在一個makefile中描述。
all: p1 p2 p3 p1:p1.c p2:p2.c p3:p3.c