您好,登錄后才能下訂單哦!
在 makefile 中是支持程序設(shè)計語言中變量的概念的,makefile 中的變量只代表文本數(shù)據(jù)(字符串)。那么在 makefile 中的變量名的規(guī)則又有哪些呢?a> 變量名可以包含字符、數(shù)字以及下劃線;b> 不能包含 ":" , "#" , "=" 或 " ";c> 變量名大小寫敏感。下來我們來看看變量的定義和使用,如下
下來我們以代碼為例來進行分析說明
CC := gcc TARGET := hello.out $(TARGET) : func.o main.o $(CC) -o $(TARGET) func.o main.o func.o : func.c $(CC) -o func.o -c func.c main.o : main.c $(CC) -o main.o -c main.c .PHONY : rebuild clean all rebuild : clean all all : $(TARGET) clean : rm *.o $(TARGET)
我們來看看編譯結(jié)果是否和之前一樣
效果和之前是一樣的,那么我們?yōu)楹芜@樣定義變量呢?在一些大型的工程項目中,我們可能要編譯好幾個版本,因此定義不同的 TARGET 便會編譯出不同的版本。而且這種方法還有個好處就是編譯器可以由我們自定義選擇,比如我們將上面的 gcc 編譯器換成 g++,則只需將 CC := gcc 換成 CC := g++,下來我們來看看編譯效果
我們看到編譯器已經(jīng)換為 g++ 了。下來我們來講講 makefile 中變量的賦值方式,分為四種:a> 簡單賦值(:=);b> 遞歸賦值(=);c> 條件賦值(?=);d> 追加賦值(+=)。那么不同的賦值方式其意義是不同的!下來我們來對這四種賦值方式進行一一的解釋說明
a> 簡單賦值(:=):程序設(shè)計語言中的通用的賦值方式,只針對當(dāng)前語句的變量是有效的。其用法如下
b> 遞歸賦值(=):賦值操作可能影響多個其他變量,所有與目標變量相關(guān)的其他變量都將受到影響。其用法如下
c> 條件賦值(?=):如果變量未定義,使用賦值符號中的值定義變量;如果變量已經(jīng)定義,則賦值無效。其用法如下
d> 追加賦值(+=):原變量值之后加上一個新值,原變量值與新值之間由空格隔開。其用法如下
下來我們還是以代碼為例來進行說明
# ex1 (:=) x := foo y := $(x)b x := new .PHONY : test test : @echo "# ex1 (:=)" @echo "x => $(x)" @echo "y => $(y)"
經(jīng)過簡單賦值后,x 就為 new 了,而 y 則應(yīng)該為 foob。我們來看看編譯效果
我們看到結(jié)果和我們所分析的是一樣的。下來我們來看看遞歸賦值
# ex2 (=) x = foo y = $(x)b x = new .PHONY : test test : @echo "# ex2 (=)" @echo "x => $(x)" @echo "y => $(y)"
我們分析下,因為遞歸賦值是與目標相關(guān)的其他變量都會受到影響,那么 x 最后應(yīng)該為 new,y 就應(yīng)該為 newb。我們來看看編譯結(jié)果
為了更加形象對遞歸賦值進行說明,我們編寫代碼如下
a = $(b) b = $(c) c = hello-makefile .PHONY : test test : @echo "# ex2 (=)" @echo "x => $(x)" @echo "y => $(y)" @echo "a => $(a)" @echo "b => $(b)" @echo "c => $(c)"
我們定義 a 依賴于 b,而 b 依賴于 c。此時 c 為 hello-makefile,那么依賴于它的 b 也就變?yōu)?hello-makefile 了,此時依賴于 b 的 a 也就變?yōu)?hello-makefile 了。x 和 y 此時沒定義,看看編譯會報錯嗎?
編譯是通過的,證明在 makefile 中是支持這樣的寫法的,x 和 y 沒定義,自然也就為空了。那么 a,b,c 和我們所分析的是一樣的。所以在 makefile 中我們一般是很少直接用遞歸賦值的,因為有可能會意想不到地改變依賴于它的目標。下來我們來看看條件賦值
# ex3 (?=) x := foo y := $(x)b x ?= new .PHONY : test test : @echo "# ex3 (?=)" @echo "x => $(x)" @echo "y => $(y)"
我們在最后對 x ?= new,意思是如果前面沒定義的話,此時它便為 new。所以 x 最后應(yīng)該為 foo,y 就應(yīng)該為 foob。我們來看看編譯結(jié)果
結(jié)果確實是這樣的,一般我們會在新定義變量時采用這樣的寫法,以防止改變之前定義時的語義。最后看看追加賦值
# ex4 (+=) x := foo y := $(x)b x += new .PHONY : test test : @echo "# ex4 (+=)" @echo "x => $(x)" @echo "y => $(y)"
根據(jù)追加賦值的定義,它是直接在后面加上,但是有一個空格。所以 x 最后應(yīng)該為 foo new,y 就應(yīng)該為 foob。我們來看看編譯結(jié)果
我們看懂啊效果確實是這樣的,通過對四種賦值方式的學(xué)習(xí),我們發(fā)現(xiàn)簡單賦值是跟 C 語言中是一樣的。追加賦值和 C++ 語言中的語義也是一樣的,一般在大型項目中是不會采用遞歸賦值這種寫法的。
歡迎大家一起來學(xué)習(xí) makefile,可以加我QQ:243343083。
免責(zé)聲明:本站發(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)容。