如何自己編寫Makefile 高階篇

2021-06-22 22:53:46 字數 3600 閱讀 4136

上篇文章介紹了makefile的編寫方法,相信大家有所收穫。本文繼續深入makefile,看看還有哪些高階用法可以大大提公升我們的工作效率。本文是接著上篇博文——如何自己編寫makefile的,因此如果沒有看過上篇文章的,建議先閱讀一遍,以便更好地理解本文。

在規則中使用萬用字元

乙個簡單的例子

clean:

rm -f *.o

表示刪除所有.o檔案,這很好理解。

print: *.c

lpr -p $?

touch print

表明print的生成依賴於所有.c檔案,$?是乙個自動化變數,以後會詳細介紹。

檔案搜尋

在一些大的工程中,有大量的原始檔,我們通常的做法是把這許多的原始檔分類,並存放在不同的目錄中。所以,當make需要去找尋檔案的依賴關係時,你可以在檔案前加上路徑,但最好的方法是把乙個路徑告訴make,讓make在自動去找。

makefile檔案中的特殊變數「vpath」就是完成這個功能的,如果沒有指明這個變數,make只會在當前的目錄中去找尋依賴檔案和目標檔案。如果定義了這個變數,那麼,make就會在當當前目錄找不到的情況下,到所指定的目錄中去找尋檔案了。

vpath = src:../headers

上面的的定義指定兩個目錄,「src」和「../headers」,make會按照這個順序進行搜尋。目錄由「冒號」分隔。

另乙個設定檔案搜尋路徑的方法是使用make的「vpath」關鍵字(注意,它是全小寫的), 這不是變數,這是乙個make的關鍵字,這和上面提到的那個vpath變數很類似,但是它更為靈活。它可以指定不同的檔案在不同的搜尋目錄中。這是乙個很 靈活的功能。它的使用方法有三種:

1、vpath

為符合模式的檔案指定搜尋目錄。

2、vpath

清除符合模式的檔案的搜尋目錄。

3、vpath

清除所有已被設定好了的檔案搜尋目錄。

vapth使用方法中的需要包含「%」字元。「%」的意思是匹 配零或若干字元,例如,「%.h」表示所有以「.h」結尾的檔案。指定了要搜尋的檔案集, 而則指定了的檔案集的搜尋的目錄。例如:

vpath %.h ../headers

該語句表示,要求make在「../headers」目錄下搜尋所有以「.h」結尾的檔案。(如果某檔案在當前目錄沒有找到的話)

我們可以連續地使用vpath語句,以指定不同搜尋策略。如果連續的vpath語句中出現了相同的,或是被重複了的,那麼,make會按照vpath語句的先後順序來執行搜尋。如:

vpath %.c foo

vpath %.c blish

vpath %.c bar

其表示「.c」結尾的檔案,先在「foo」目錄,然後是「blish」,最後是「bar」目錄。

vpath %.c foo:bar

vpath %.c blish

而上面的語句則表示「.c」結尾的檔案,先在「foo」目錄,然後是「bar」目錄,最後才是「blish」目錄。

偽目標

最早先的乙個例子中,我們提到過乙個「clean」的目標,這是乙個「偽目標」,

clean:

rm *.o temp

正像我們前面例子中的「clean」一樣,即然我們生成了許多檔案編譯檔案,我們也應該提供乙個清除它們的「目標」以備完整地重編譯而用。 (以「make clean」來使用該目標)

因為,我們並不生成「clean」這個檔案。「偽目標」並不是乙個檔案,只是乙個標籤,由於 「偽目標」不是檔案,所以make無法生成它的依賴關係和決定它是否要執行。我們只有通過顯示地指明這個「目標」才能讓其生效。當然,「偽目標」的取名不 能和檔名重名,不然其就失去了「偽目標」的意義了。

當然,為了避免和檔案重名的這種情況,我們可以使用乙個特殊的標記「.phony」來顯示地指明乙個目標是「偽目標」,向make說明,不管是否有這個檔案,這個目標就是「偽目標」。

.phony : clean

只要有這個宣告,不管是否有「clean」檔案,要執行「clean」這個目標,只有「make clean」這樣。於是整個過程可以這樣寫:

.phony: clean

clean:

rm *.o temp

偽目標一般沒有依賴的檔案。但是,我們也可以為偽目標指定所依賴的檔案。偽目標同樣可以作為 「預設目標」,只要將其放在第乙個。乙個示例就是,如果你的makefile需要一口氣生成若干個可執行檔案,但你只想簡單地敲乙個make完事,並且, 所有的目標檔案都寫在乙個makefile中,那麼你可以使用「偽目標」這個特性:

all : prog1 prog2 prog3

.phony : all

prog1 : prog1.o utils.o

cc -o prog1 prog1.o utils.o

prog2 : prog2.o

cc -o prog2 prog2.o

prog3 : prog3.o sort.o utils.o

cc -o prog3 prog3.o sort.o utils.o

我們知道,makefile中的第乙個目標會被作為其預設目標。我們宣告了乙個「all」的偽 目標,其依賴於其它三個目標。由於偽目標的特性是,總是被執行的,所以其依賴的那三個目標就總是不如「all」這個目標新。所以,其它三個目標的規則總是 會被決議。也就達到了我們一口氣生成多個目標的目的。「.phony : all」宣告了「all」這個目標為「偽目標」。

隨便提一句,從上面的例子我們可以看出,目標也可以成為依賴。所以,偽目標同樣也可成為依賴。看下面的例子:

.phony: cleanall cleanobj cleandiff

cleanall : cleanobj cleandiff

rm program

cleanobj :

rm *.o

cleandiff :

rm *.diff

「make cleanall」將清除所有要被清除的檔案。「cleanobj」和「cleandiff」這兩個偽目標有點像「子程式」的意思。我們可以輸入「make cleanall」和「make cleanobj」和「make cleandiff」命令來達到清除不同種類檔案的目的。

makefile中的自動化變數 

$@代表法定中的目標檔名。萬一目標是乙個文件(linux中,等閒稱.a檔案為文件),那麼它代表這個文件的檔名。在多目標的形式法定中,它代表的是哪個引發法定被厲行的目標檔名。

$%法定的目標檔案是乙個靜態庫檔案時,代表靜態庫的乙個成員名。例如,法定的目標是「foo.a(bar.o)」,那麼,「$%」的值就為「bar.o」,「$@」的值為「foo.a」。萬一目標不是函式庫檔案,其值為空。

$<

法定的第乙個依靠檔名。萬一是隱含法定,則它代表穿越目標指定的第乙個依靠檔案。

$?所有比目標檔案更新的依靠檔案列表,空格瓜分。萬一目標是靜態庫檔名,代表的是庫成

員(.o檔案)的更新情形。

$^法定的所有依靠檔案列表,利用空格分隔。萬一目標是靜態庫檔名,它所代表的只能是所有庫成員(.o檔案)名。乙個檔案可重複的展目前目標的依靠中,變數「$^」只登記它的順次引用情形。即便說變數「$^」會去掉重複的依靠檔案。

$+相仿「$^」,然而它保留了依靠檔案中重複揭示的檔案。重要用在過程鏈結時,庫的穿插引用處所。

更多的makefile知識

學到這,我覺得寫出一般的makefile是沒有任何問題了。

自己編寫Makefile檔案

首先,編寫makefile檔案,要明白一些概念。很多的windows程式設計師都不知道這個東西,因為windows的ide編譯環境都打好包做了這個工作了。而在linux下編譯自己的程式,學會寫makefile是做為linux程式的基礎。無論是哪種編譯環境下 c c pas 首先都是先把原始檔編譯成中...

如何編寫寫簡單的Makefile

在linux系統中,當需要編譯多個.c檔案時,如果我們乙個乙個去編譯原始檔,會變得十分麻煩,所以我們通常我編寫乙個makefile檔案,這個檔案相當於乙個指令碼檔案,當完成makefile檔案的編寫後,只需要通過make命令就可以完成原始檔的預編譯,編譯,彙編,鏈結過程,生成可執行檔案。target...

如何編寫簡單的Makefile檔案

1.gnu make是乙個命令工具,是乙個用來控制軟體構建過程的自動化管理工具。make工具通過稱為makefile的檔案完成並自動維護編譯工作。3 makefile定義了一系列的規則來指定,乙個工程中的哪些檔案需要先編譯,哪些檔案需要後編譯,哪些檔案需要重新編譯,甚至於進行更複雜的功能操作。總之,...