我們在前面學(xué)習(xí)了 makefile 中的相關(guān)知識,今天我們來看看在 makefile 中變量值的替換。它的替換是指使用指定字符(串)替換變量值中的后綴字符(串),語法格式為:$(var:a=b) 或 ${var:a=b}。注意:a> 替換表達式中不能有任何的空格;b> make 中支持使用 ${ } 對變量進行取值。格式如下
你所需要的網(wǎng)站建設(shè)服務(wù),我們均能行業(yè)靠前的水平為你提供.標(biāo)準(zhǔn)是產(chǎn)品質(zhì)量的保證,主要從事網(wǎng)站制作、網(wǎng)站設(shè)計、企業(yè)網(wǎng)站建設(shè)、成都手機網(wǎng)站制作、網(wǎng)頁設(shè)計、成都品牌網(wǎng)站建設(shè)、網(wǎng)頁制作、做網(wǎng)站、建網(wǎng)站。創(chuàng)新互聯(lián)擁有實力堅強的技術(shù)研發(fā)團隊及素養(yǎng)的視覺設(shè)計專才。
還有種便是變量的模式替換是指使用 % 保留變量值中的指定字符,替換其他字符。語法格式為:$(var:a%b=x%y) 或 ${var:a%b=x%y}。注意:a> 替換表達式中不能有任何的空格;b> make 中支持使用 ${ } 對變量進行取值。格式如下
規(guī)則中的模式替換如下
它的意義是通過 target-pattern 從 targets 中匹配子目標(biāo);再通過 prereq-pattern 從子目標(biāo)生成依賴;進而構(gòu)成完整的規(guī)則。我們來看看規(guī)則中的模式替換示例如下
下來我們通過代碼來分析說明
src1 := a.cc b.cc c.cc obj1 := $(src1:cc=o) test1 : @echo "obj1 => $(obj1)" src2 := a11b.c a22b.c a33b.c obj2 := $(src2:a%b.c=x%y) test2 : @echo "obj2 => $(obj2)"
我們根據(jù)之前說的,在 obj1 中將會把 .cc 替換成 .o,把 obj2 中的 a11b.c a22b.c a33b.c 替換成 x11y x22y x33y。我們來看看編譯器效果
我們看到結(jié)果和我們分析的是一樣的。下來再來看看模式替換,將之前的 makefile 進行改編
CC := g++ TARGET := hello-makefile.out OBJS := func.o main.o $(TARGET) : $(OBJS) $(CC) -o $@ $^ $(OBJS) : %.o : %.c $(CC) -o $@ -c $^ .PHONY : rebuild clean all rebuild : clean all all : $(TARGET) clean : $(RM) *.o $(TARGET)
我們來看看編譯效果和之前的是一樣的嗎?
結(jié)果是一樣的,這樣的寫的意義在哪呢?在大型的工程項目中,.c源文件是成千上萬的。我們就可以利用模式替換來代替重復(fù)的工作,比如我們想添加一個 const.c 文件,便可以直接在第 3 行直接加上 const.o 就OK了。我們來試試看
const.c 源碼
const char* g_hello = "hello makefile";
func.c 源碼
#include "stdio.h" extern char* g_hello; void foo() { printf("void foo() : %s\n", g_hello); }
main.c 源碼
extern void foo(); int main() { foo(); return 0; }
我們來看看編譯效果
我們看到在編譯的時候自動加上了編譯 const.c 的命令,并最終正確打印出結(jié)果。這樣感覺是不是很方便哈。那么在 makefile 中變量值還可以嵌套引用,就是一個變量名之中可以包含對其它變量的引用,嵌套引用的本質(zhì)是使用一個變量來表示另外一個變量。格式如下
下來我們來說說命令行變量,在運行 make 時直接在命令行定義變量。命令行變量默認覆蓋 makefile 中定義的變量,格式如下
那么命令行變量可以覆蓋 makefile 中定義的變量,如果我們不小心手誤覆蓋了呢?這時 override關(guān)鍵字就登場了。它是用于指示 makefile 中定義的變量不能被覆蓋,變量的定義個賦值都需要使用 override 關(guān)鍵字。格式如下
下來我們來看看 makefile 中的 define關(guān)鍵字,它是用于在 makefile 中定義多行變量,多行變量的定義從變量名開始帶 endef 結(jié)束??墒褂?override 關(guān)鍵字防止變量被覆蓋,define 定義的變量等價于使用 = 定義的變量。格式入下
下來我們還是以代碼為例來進行說明
hm := hello makefile override var := override-test define foo I'm fool! endef override define cmd @echo "run cmd ls ..." @ls endef test : @echo "hm => $(hm)" @echo "var => $(var)" @echo "foo => $(foo)" ${cmd}
我們來編譯看看結(jié)果
我們看到在沒有被 override 關(guān)鍵字修飾的變量 hm 可以在命令行對它進行改寫,但是變量 cmd 因為被 override 修飾了,因此就在命令行里面的修改是無效的。下來我們還說說 makefile 中的環(huán)境變量(全局變量),在 makefile 中能夠直接使用環(huán)境變量的值。它是在定義了同名變量的話,環(huán)境變量將被覆蓋,運行 make 時指定“-e”選項,優(yōu)先使用環(huán)境變量。那么為什么要在 makefile 中使用環(huán)境變量呢?它的優(yōu)勢是環(huán)境變量可以在所有的 makefile 中使用,劣勢是過多的依賴于環(huán)境變量便會導(dǎo)致移植性降低。那么變量在不同的 makefile 支架的傳遞方式有哪些呢?a> 直接在外部定義環(huán)境變量進行傳遞;b> 使用 export 定義變量進行傳遞;c> 定義 make 命令進行傳遞(一般推薦使用這種)。
下來我們還是以代碼為例來進行分析
export var := D.T.Software new := TDelphi test : @echo "make another file ..." @$(MAKE) -f makefile.4 @$(MAKE) -f makefile.4 new:=$(new)
makefile.4 源碼
test : @echo "var => $(var)" @echo "new => $(new)"
我們來看看編譯結(jié)果
我們看到在第一次的時候 new 為空,在第二次的時候 new 為我們設(shè)置的字符串。兩次的 var 都傳遞過去了就是因為我們使用了 export 這個關(guān)鍵字。下來我們來看看目標(biāo)變量(局部變量),其作用域只在指定目標(biāo)及連帶規(guī)則中。格式如下
那么模式變量便是目標(biāo)變量的擴展,其作用域只在符合模式的目標(biāo)及連帶規(guī)則中。格式如下
下來還是以代碼為例來進行分析說明
var := D.T.Software new := TDelphi test : var := test-var %e : override new := test-new test : another @echo "test :" @echo "var => $(var)" @echo "new => $(new)" another : @echo "another :" @echo "var => $(var)" @echo "new => $(new)" rule : @echo "rule :" @echo "var => $(var)" @echo "new => $(new)"
我們來看看編譯結(jié)果
因為 %e 的匹配規(guī)則,所以在目標(biāo) rule 中,它的 new 為 test-new,凡是跟 test 目標(biāo)相關(guān)的 var 都是 test-var。通過對 makefile 中變量的學(xué)習(xí),總結(jié)如下:1、makefile 中的變量值能夠嵌套引用;2、命令行中定義的變量能夠覆蓋 makefile 中定義的變量;3、override 用于提示 makefile 中定義的變量不能被覆蓋;4、define 用于在 makefile 中定義值為多行的變量;5、makefile 中的三種變量:a> 全局變量是指 makefile 外部定義的環(huán)境變量;b> 文件變量是在 makefile 中定義的變量;c> 局部變量是指定目標(biāo)的變量。
歡迎大家一起來學(xué)習(xí) makefile,可以加我QQ:243343083。