.PHONY makefile中的僞目標

我的理解:

  拿clean舉例,如果make完成後,自己另外定義一個名叫clean的文件,再執行make clean時,將不會執行rm命令。 

  爲了避免出現這個問題,需要.PHONY: clean

 

=======================================================================================

所謂僞目標就是這樣一個目標,它不代表一個真正的文件名,在執行make時可以指定這個目標來執行其所在規則定義的命令,有時我們將一個僞目標成爲標籤。

那麼到底什麼是僞目標呢?可能作爲初學者還不會在乎這個問題,下面我們來看下我們將在什麼時候需要它。

首先來看下面一個例子:

當前目錄下只有一個myls1.c,於是爲了讓程序讓makefile來管理,寫了一個如下的簡單的makefile。

執行:

大家會發現,真的可以利用這個makefile管理當前的工程,也能如期按照我們的要求生成執行文件myls。

執行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

如果這樣寫,會出現幾個問題:

1、 當子目錄執行make出現錯誤,make不會退出;

2、 使用這種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

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