make--僞目標 、不同的賦值方式、變量

一.僞目標的引入

a.默認情況下
1.make認爲目標對應一個文件
2.make比較目標文件和依賴文件的新舊關係,決定是否執行命令
3.make以文件處理作爲第一優先級
示例

hello.out all : func.o main.o
    gcc -o hello.out func.o main.o

func.o : func.c
    gcc -o func.o -c func.c

main.o : main.c
    gcc -o main.o -c main.c

clean :
    rm *.o hello.out

該示例與之前的示例加上了clean
make--僞目標 、不同的賦值方式、變量
make--僞目標 、不同的賦值方式、變量
由上面的圖可以看出,在第二個我們假設在目錄下存在一個clean文件(可以通過touch命令來創建),如果我們此時運行make clean,會發現make總是提示clean文件是最新的,make這種行爲從原理上來說是因爲它將clean當作文件來處理,由於在當前目錄下找到了這個文件,加上clean目標沒有任何先覺條件,所以當要求make爲我們創建clean目標時它就會認爲clean是最新的,從而“拒絕”進行文件的清除操作。
出現這種情形,是因爲我們對於clean目標的定義與make所理解的有所出入。目標文件名與Makefile中的目標名重名在現實項目中是難免的,所以假目標的概念被提出
makefile中的僞目標
1.通過.PHONY關鍵字聲明一個僞目標
2.僞目標不對應任何實際的文件
3.不管僞目標的依賴是否更新,命令總是執行
僞目標的語法:先聲明,後使用
make--僞目標 、不同的賦值方式、變量
加入.PHONY的示例

.PHONY: clean
hello.out all : func.o main.o
    gcc -o hello.out func.o main.o

func.o : func.c
    gcc -o func.o -c func.c

main.o : main.c
    gcc -o main.o -c main.c

clean :
    rm *.o hello.out

運行結果如圖所示
make--僞目標 、不同的賦值方式、變量
如圖所示,我們可以看出在當前目錄下即使存在clean文件,有了僞目標.PHONY的使用運行make clean命令會執行刪除操作

二.不同的賦值方式與變量

1.makefile中支持程序設計語言中變量的概念
2.makefile中的變量只代表文本數據(字符串)
3.makefile中的變量名規則
(變量名可以包含字符、數字、下劃線;不能包含“:”,“#”,“=”或“ ”;變量名大小寫敏感)
A.變量的定義和使用(通過使用變量可以使得Makefile更具有可維護性)
make--僞目標 、不同的賦值方式、變量

.PHONY: clean

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)

運行結果如圖示
make--僞目標 、不同的賦值方式、變量
可以看到在加入了變量之後,生成的結果是一樣的
B.makefile中變量的賦值方式
1.簡單賦值(:=)
2.遞歸賦值(=)
3.條件賦值(?=)
4.追加賦值(+=)
a.簡單賦值(:=)
1.程序設計語言中的通用的賦值方式
2.只針對當前語句的變量有效

.PHONY: all

x=foo
y=$(x)b
x=later

xx:=foo
yy:=$(xx)b
ixx:=later

all:
    @echo "xx=$(xx),yy=$(yy)"

結果如圖
make--僞目標 、不同的賦值方式、變量
b.遞歸賦值(=)
1.賦值操作可能影響多個其他變量
2.所有與目標變量相關的其他變量都將受到影響

.PHONY: all

foo=$(bar)
bar=$(ugh)
ugh=MYLOVE

all:
    @echo $(foo)

結果如圖
make--僞目標 、不同的賦值方式、變量
c.條件賦值(?=)
1.如果變量未定義,使用賦值符號中的值定義變量
2.如果變量已經定義,則賦值無效

.PHONY: all

x:=foo
y:=$(x)b
x?=new

all:
    @echo "x=>$(x)"
    @echo "y=>$(y)"

make--僞目標 、不同的賦值方式、變量
d.追加賦值(+=)
1.原變量值之後加上一個新值
2.原變量值與新值之間由空格隔開

.PHONY: all

x:=foo
y:=$(x)b
x+= $(y)

all:
    @echo "x=>$(x)"
    @echo "y=>$(y)"

make--僞目標 、不同的賦值方式、變量

三.預定義變量的使用

1.$@ 用於表示一個規則的目標 當一個規則中有多個目標時 $@所指的是其中任一造成規則命令被運行的目標
2.$^ 表示的是規則中的所有先決條件
3.$< 表示的是規則中的第一個先決條件

運用這些規則對之前的代碼進行修改


CC := g++
TARGET := hello.out

$(TARGET) : func.o main.o
    $(CC) -o $@ $^

func.o : func.c
    $(CC) -o $@ -c $^

main.o : main.c
    $(CC) -o $@ -c $^

.PHONY : rebuild clean all

rebuild : clean all

all : $(TARGET)

clean :
    $(RM) *.o $(TARGET)

結果如圖所示
make--僞目標 、不同的賦值方式、變量
這樣寫的原因在之前就已經提到使得Makefile更具有可維護性

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章