溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Linux下Makefile怎么用

發(fā)布時間:2021-12-13 11:08:04 來源:億速云 閱讀:187 作者:小新 欄目:開發(fā)技術(shù)

這篇文章主要為大家展示了“Linux下Makefile怎么用”,內(nèi)容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學習一下“Linux下Makefile怎么用”這篇文章吧。

Makefile

一個工程文件中的源文件可能有很多,并且不同的功能、模塊等都放在不同的目錄中,常規(guī)的編譯已經(jīng)不能高效化的處理這樣的問題,而Makefile就是為解決這一問題而來。

Makefile一旦寫好,只需一個make指令,即可完成Makefile文件中所編寫的所有指令,從而編譯整個工程文件,極大的提高了效率。

make是一個命令工具,用來解釋Makefile中的命令。

Makefile文件命名和規(guī)則

文件命名

采用makefile或Makefile都可。

Makefile規(guī)則

Makefile中的命令規(guī)則如下:

xxx(目標文件):xxx(依賴文件)
制表符)命令(shell命令)
其中,目標文件即最終要生成的文件(偽目標除外),依賴文件即生成目標文件所需的文件,命令即shell命令。

注意,命令前必須有一個tab縮進。

例如:

#Makefile
app: a.c b.c #目標:依賴
	gcc a.c b.c -o app #注意這行最開始的縮進

make以上這個Makefile后就會將目錄下的a.cb.c編譯為目標文件app。

Makefile的工作原理

Makefile中的命令在執(zhí)行前,會檢查是否存在所需的依賴文件

如果存在:執(zhí)行命令

如果不存在:向下檢查其他規(guī)則,是否存在其他規(guī)則生成當前規(guī)則所需要的依賴,如果有,則執(zhí)行該規(guī)則中的命令。

例如:

#Makefile
app: a.o b.o
	gcc a.o b.o -o app
a.o: a.c
	gcc -c a.c -o a.o
b.o: b.c
	gcc -c b.c -o b.o

在上方這個Makefile中,當執(zhí)行到app規(guī)則時,會發(fā)現(xiàn)所需的依賴文件a.o與b.o都不存在于當前目錄,所以會向下尋找是否有其他規(guī)則生成此文件,當尋找到a.o規(guī)則時,發(fā)現(xiàn)其是所需的文件,就執(zhí)行g(shù)cc -c a.c -o a.o,b.o同理。

Makefile在執(zhí)行規(guī)則中的命令時,會比較目標文件和依賴文件的修改時間
如果依賴文件晚于目標文件修改時間,即依賴文件在上一次生成目標后進行過修改,則會重新生成目標文件。
如果依賴文件早于目標文件修改時間,即依賴文件在上一次生成目標后沒進行修改,則不會執(zhí)行相應的命令。
例如,你對一個Makefile使用兩次make,第二次會提示make:"app"已是最新。

利用這個特性,在加上我們將依賴與目標分級生成,即上方第二個Makefile,這樣當我們僅修改其中的a.c文件,再一次make只會執(zhí)行a.o規(guī)則與app規(guī)則,b.o規(guī)則因為b.c未修改而不執(zhí)行,這樣可以大大減少資源浪費。

Makefile變量

以上雖然可以減少編譯代碼的重復量,但是如果一個工程中有1000個.c .h文件,我們編寫一個Makefile就會浪費大量時 間。因此,我們要采用一些變量來提高效率。

變量的獲取
我們使用 $(變量名) 來使用變量。

自定義變量
我們使用 變量名 = 變量值var = hello來自定義我們所需的變量。
例如上方第一個Makefile就可改寫為:

#Makefile
rsc = a.c b.c
app: $(rsc) #目標:依賴
	gcc $(rsc) -o app #注意這行最開始的縮進

預定義的變量
有部分變量是系統(tǒng)預定義的,我們可以直接使用。
AR:歸檔維護程序的名稱,默認值為ar
CC:C編譯器的名稱,默認值為cc
CXX:C++編譯器的名稱,默認值為g++
$@:目標的完整名稱
$<:第一個依賴文件的名稱
$^:所有依賴文件的名稱

為了方便理解接下來的例子,我們簡單講解一下Makefile中的模式匹配。
%.o:%.c 中,%是 通配符,匹配一個字符串,而兩個%則匹配同一個字符串。
例如上方第二個Makefile可改寫為:

#Makefile
rcs = a.o b.o
app: $(rcs)
	$(CC) $(rcs) -o $@
%.o: %.c #上方規(guī)則會執(zhí)行兩次此規(guī)則
	$(CC) -c $< -o $@

Makefile函數(shù)

我們可以看到,上面這個Makefile已經(jīng)相對簡單了,但是,還是沒有解決工程中文件很多的情況,rcs的獲取還是要我們輸入每個需要編譯的文件,那么,就要采用函數(shù)來替我們?nèi)懭脒@些依賴文件。

$(wildcard PATTERN. . .)
這個函數(shù)的功能是獲取指定目錄下指定類型的文件。
其中參數(shù)PATTERN是某個目錄下某種類型的文件,多個目錄多個類型可用空格分隔。
返回值是一個若干個文件的文件列表,文件名用空格隔開。

例如:

$(wildcard ./*.c) 返回當前目錄下的所有以c為后綴的文件。

$(patsubst pattern, replacement, text)
這個函數(shù)的功能是查找text中的單詞是否符合模式pattern,如果符合,則用replacement替換。
pattern可以包括通配符 % 。如果replacement中也包含 % ,那么replacement中的 % 將和 pattern中的 % 保持一致。
返回值為替換后的字符串。

例如:

$(patsubst %.c, %.o, a.c, b.c) 返回a.o, b.o。

這樣,我們上面那個例子就可以改寫為:

#Makefile
rcs = $(wildcard ./*.c)
objs = (patsubst %.c, %.o, $(src))
app: $(objs)
	$(CC) $(objs) -o $@
%.o: %.c #上方規(guī)則會執(zhí)行兩次此規(guī)則
	$(CC) -c $< -o $@

Makefile clean規(guī)則

在我們執(zhí)行完make指令后,會發(fā)現(xiàn)當前目錄下多出了很多以o為后綴的文件,但是我們僅需要最終的目標文件app,其他的都是多余的,我們該如何處理。clean規(guī)則就會幫助我們處理他們。

clean

我們只用將clean規(guī)則添加到Makefile的最后,即可在每次編譯完成后執(zhí)行clean規(guī)則中的命令。如:

#Makefile
rcs = $(wildcard ./*.c)
objs = (patsubst %.c, %.o, $(src))
app: $(objs)
	$(CC) $(objs) -o $@
%.o: %.c #上方規(guī)則會執(zhí)行兩次此規(guī)則
	$(CC) -c $< -o $@
clean:
	rm $(objs) -f #rm指令刪除 -f迭代刪除

但是你會發(fā)現(xiàn)當前目錄下多出了一個clean目標文件,依舊會采用Makefile的策略,對比修改時間,導致我們時常及時執(zhí)行了clean,還是無法清除文件,那么,我們就需要接下來這個操作。

我們將clean定義為偽目標,即 .PHONY:clean 那么它就不會生成目標文件,少了對比,那么每次都會執(zhí)行。

例如:

#Makefile
rcs = $(wildcard ./*.c)
objs = (patsubst %.c, %.o, $(src))
app: $(objs)
	$(CC) $(objs) -o $@
%.o: %.c #上方規(guī)則會執(zhí)行兩次此規(guī)則
	$(CC) -c $< -o $@
.PHONY: clean #偽目標
clean:
	rm $(objs) -f #rm指令刪除 -f迭代刪除

以上是“Linux下Makefile怎么用”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學習更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI