1. 程式人生 > >變量和不同的賦值方式(四)

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

makefile 簡單賦值(:=) 遞歸賦值(=) 條件賦值(?=) 追加賦值(+=)

在 makefile 中是支持程序設計語言中變量的概念的,makefile 中的變量只代表文本數據(字符串)。那麽在 makefile 中的變量名的規則又有哪些呢?a> 變量名可以包含字符、數字以及下劃線;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)

我們來看看編譯結果是否和之前一樣

技術分享圖片

效果和之前是一樣的,那麽我們為何這樣定義變量呢?在一些大型的工程項目中,我們可能要編譯好幾個版本,因此定義不同的 TARGET 便會編譯出不同的版本。而且這種方法還有個好處就是編譯器可以由我們自定義選擇,比如我們將上面的 gcc 編譯器換成 g++,則只需將 CC := gcc 換成 CC := g++,下來我們來看看編譯效果

技術分享圖片

我們看到編譯器已經換為 g++ 了。下來我們來講講 makefile 中變量的賦值方式,分為四種:a> 簡單賦值(:=);b> 遞歸賦值(=);c> 條件賦值(?=);d> 追加賦值(+=)

。那麽不同的賦值方式其意義是不同的!下來我們來對這四種賦值方式進行一一的解釋說明

a> 簡單賦值(:=):程序設計語言中的通用的賦值方式,只針對當前語句的變量是有效的。其用法如下

技術分享圖片

b> 遞歸賦值(=):賦值操作可能影響多個其他變量,所有與目標變量相關的其他變量都將受到影響。其用法如下

技術分享圖片

c> 條件賦值(?=):如果變量未定義,使用賦值符號中的值定義變量;如果變量已經定義,則賦值無效。其用法如下

技術分享圖片

d> 追加賦值(+=):原變量值之後加上一個新值,原變量值與新值之間由空格隔開。其用法如下

技術分享圖片

下來我們還是以代碼為例來進行說明

# ex1 (:=)
x := foo
y := $(x)b
x := new

.PHONY : test

test :
    @echo "# ex1 (:=)"
    @echo "x => $(x)"
    @echo "y => $(y)"

經過簡單賦值後,x 就為 new 了,而 y 則應該為 foob。我們來看看編譯效果

技術分享圖片

我們看到結果和我們所分析的是一樣的。下來我們來看看遞歸賦值

# ex2 (=)
x = foo
y = $(x)b
x = new

.PHONY : test

test :
    @echo "# ex2 (=)"
    @echo "x => $(x)"
    @echo "y => $(y)"

我們分析下,因為遞歸賦值是與目標相關的其他變量都會受到影響,那麽 x 最後應該為 new,y 就應該為 newb。我們來看看編譯結果

技術分享圖片

為了更加形象對遞歸賦值進行說明,我們編寫代碼如下

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 也就變為 hello-makefile 了,此時依賴於 b 的 a 也就變為 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 最後應該為 foo,y 就應該為 foob。我們來看看編譯結果

技術分享圖片

結果確實是這樣的,一般我們會在新定義變量時采用這樣的寫法,以防止改變之前定義時的語義。最後看看追加賦值

# ex4 (+=)
x := foo
y := $(x)b
x += new

.PHONY : test

test :
    @echo "# ex4 (+=)"
    @echo "x => $(x)"
    @echo "y => $(y)"

根據追加賦值的定義,它是直接在後面加上,但是有一個空格。所以 x 最後應該為 foo new,y 就應該為 foob。我們來看看編譯結果

技術分享圖片

我們看懂啊效果確實是這樣的,通過對四種賦值方式的學習,我們發現簡單賦值是跟 C 語言中是一樣的。追加賦值和 C++ 語言中的語義也是一樣的,一般在大型項目中是不會采用遞歸賦值這種寫法的。


歡迎大家一起來學習 makefile,可以加我QQ:243343083

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