溫馨提示×

溫馨提示×

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

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

變量和不同的賦值方式(四)

發(fā)布時間:2020-02-28 03:44:20 來源:網(wǎng)絡(luò) 閱讀:568 作者:上帝之子521 欄目:系統(tǒng)運維

        在 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。

向AI問一下細節(jié)

免責(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)容。

AI