跟我一起写 Makefile(八)
六、多行變量
?
還有一種設(shè)置變量值的方法是使用define關(guān)鍵字。使用define關(guān)鍵字設(shè)置變量的值可以有換行,這有利于定義一系列的命令(前面我們講過“命令包”的技術(shù)就是利用這個(gè)關(guān)鍵字)。
define指示符后面跟的是變量的名字,而重起一行定義變量的值,定義是以endef關(guān)鍵字結(jié)束。其工作方式和“=”操作符一樣。變量的值可以包含函數(shù)、命令、文字,或是其它變量。因?yàn)槊钚枰訹Tab]鍵開頭,所以如果你用define定義的命令變量中沒有以[Tab]鍵開頭,那么make就不會(huì)把其認(rèn)為是命令。
下面的這個(gè)示例展示了define的用法:
??? define two-lines
??? echo foo
??? echo $(bar)
??? endef
七、環(huán)境變量
make運(yùn)行時(shí)的系統(tǒng)環(huán)境變量可以在make開始運(yùn)行時(shí)被載入到Makefile文件中,但是如果Makefile中已定義了這個(gè)變量,或是這個(gè)變量由make命令行帶入,那么系統(tǒng)的環(huán)境變量的值將被覆蓋。(如果make指定了“-e”參數(shù),那么,系統(tǒng)環(huán)境變量將覆蓋Makefile中定義的變量)
因此,如果我們?cè)诃h(huán)境變量中設(shè)置了“CFLAGS”環(huán)境變量,那么我們就可以在所有的Makefile中使用這個(gè)變量了。這對(duì)于我們使用統(tǒng)一的編譯參數(shù)有比較大的好處。如果Makefile中定義了CFLAGS,那么則會(huì)使用Makefile中的這個(gè)變量,如果沒有定義則使用系統(tǒng)環(huán)境變量的值,一個(gè)共性和個(gè)性的統(tǒng)一,很像“全局變量”和“局部變量”的特性。
當(dāng)make嵌套調(diào)用時(shí)(參見前面的“嵌套調(diào)用”章節(jié)),上層Makefile中定義的變量會(huì)以系統(tǒng)環(huán)境變量的方式傳遞到下層的Makefile中。當(dāng)然,默認(rèn)情況下,只有通過命令行設(shè)置的變量會(huì)被傳遞。而定義在文件中的變量,如果要向下層Makefile傳遞,則需要使用exprot關(guān)鍵字來聲明。(參見前面章節(jié))
當(dāng)然,我并不推薦把許多的變量都定義在系統(tǒng)環(huán)境中,這樣,在我們執(zhí)行不用的Makefile時(shí),擁有的是同一套系統(tǒng)變量,這可能會(huì)帶來更多的麻煩。
八、目標(biāo)變量
前面我們所講的在Makefile中定義的變量都是“全局變量”,在整個(gè)文件,我們都可以訪問這些變量。當(dāng)然,“自動(dòng)化變量”除外,如“$<”等這種類量的自動(dòng)化變量就屬于“規(guī)則型變量”,這種變量的值依賴于規(guī)則的目標(biāo)和依賴目標(biāo)的定義。
當(dāng)然,我樣同樣可以為某個(gè)目標(biāo)設(shè)置局部變量,這種變量被稱為“Target-specific Variable”,它可以和“全局變量”同名,因?yàn)樗淖饔梅秶辉谶@條規(guī)則以及連帶規(guī)則中,所以其值也只在作用范圍內(nèi)有效。而不會(huì)影響規(guī)則鏈以外的全局變量的值。
其語法是:
??? <target ...> : <variable-assignment>
??? <target ...> : overide <variable-assignment>
<variable-assignment>可以是前面講過的各種賦值表達(dá)式,如“=”、“:=”、“+=”或是“?=”。第二個(gè)語法是針對(duì)于make命令行帶入的變量,或是系統(tǒng)環(huán)境變量。
這個(gè)特性非常的有用,當(dāng)我們?cè)O(shè)置了這樣一個(gè)變量,這個(gè)變量會(huì)作用到由這個(gè)目標(biāo)所引發(fā)的所有的規(guī)則中去。如:
??? prog : CFLAGS = -g
??? prog : prog.o foo.o bar.o
??????????? $(CC) $(CFLAGS) prog.o foo.o bar.o
??? prog.o : prog.c
??????????? $(CC) $(CFLAGS) prog.c
??? foo.o : foo.c
??????????? $(CC) $(CFLAGS) foo.c
??? bar.o : bar.c
??????????? $(CC) $(CFLAGS) bar.c
?
在這個(gè)示例中,不管全局的$(CFLAGS)的值是什么,在prog目標(biāo),以及其所引發(fā)的所有規(guī)則中(prog.o foo.o bar.o的規(guī)則),$(CFLAGS)的值都是“-g”
九、模式變量
在GNU的make中,還支持模式變量(Pattern-specific Variable),通過上面的目標(biāo)變量中,我們知道,變量可以定義在某個(gè)目標(biāo)上。模式變量的好處就是,我們可以給定一種“模式”,可以把變量定義在符合這種模式的所有目標(biāo)上。
我們知道,make的“模式”一般是至少含有一個(gè)“%”的,所以,我們可以以如下方式給所有以[.o]結(jié)尾的目標(biāo)定義目標(biāo)變量:
??? %.o : CFLAGS = -O
同樣,模式變量的語法和“目標(biāo)變量”一樣:
??? <pattern ...> : <variable-assignment>
??? <pattern ...> : override <variable-assignment>
override同樣是針對(duì)于系統(tǒng)環(huán)境傳入的變量,或是make命令行指定的變量。
?
使用條件判斷
——————
使用條件判斷,可以讓make根據(jù)運(yùn)行時(shí)的不同情況選擇不同的執(zhí)行分支。條件表達(dá)式可以是比較變量的值,或是比較變量和常量的值。
一、示例
下面的例子,判斷$(CC)變量是否“gcc”,如果是的話,則使用GNU函數(shù)編譯目標(biāo)。
??? libs_for_gcc = -lgnu
??? normal_libs =
??? foo: $(objects)
??? ifeq ($(CC),gcc)
??????????? $(CC) -o foo $(objects) $(libs_for_gcc)
??? else
??????????? $(CC) -o foo $(objects) $(normal_libs)
??? endif
可見,在上面示例的這個(gè)規(guī)則中,目標(biāo)“foo”可以根據(jù)變量“$(CC)”值來選取不同的函數(shù)庫來編譯程序。
我們可以從上面的示例中看到三個(gè)關(guān)鍵字:ifeq、else和endif。ifeq的意思表示條件語句的開始,并指定一個(gè)條件表達(dá)式,表達(dá)式包含兩個(gè)參數(shù),以逗號(hào)分隔,表達(dá)式以圓括號(hào)括起。else表示條件表達(dá)式為假的情況。endif表示一個(gè)條件語句的結(jié)束,任何一個(gè)條件表達(dá)式都應(yīng)該以endif結(jié)束。
當(dāng)我們的變量$(CC)值是“gcc”時(shí),目標(biāo)foo的規(guī)則是:
??? foo: $(objects)
??????????? $(CC) -o foo $(objects) $(libs_for_gcc)
而當(dāng)我們的變量$(CC)值不是“gcc”時(shí)(比如“cc”),目標(biāo)foo的規(guī)則是:
??? foo: $(objects)
??????????? $(CC) -o foo $(objects) $(normal_libs)
當(dāng)然,我們還可以把上面的那個(gè)例子寫得更簡(jiǎn)潔一些:
??? libs_for_gcc = -lgnu
??? normal_libs =
??? ifeq ($(CC),gcc)
????? libs=$(libs_for_gcc)
??? else
????? libs=$(normal_libs)
??? endif
??? foo: $(objects)
??????????? $(CC) -o foo $(objects) $(libs)
二、語法
條件表達(dá)式的語法為:
??? <conditional-directive>
??? <text-if-true>
??? endif
以及:
??? <conditional-directive>
??? <text-if-true>
??? else
??? <text-if-false>
??? endif
其中<conditional-directive>表示條件關(guān)鍵字,如“ifeq”。這個(gè)關(guān)鍵字有四個(gè)。
第一個(gè)是我們前面所見過的“ifeq”
??? ifeq (<arg1>, <arg2>)
??? ifeq '<arg1>' '<arg2>'
??? ifeq "<arg1>" "<arg2>"
??? ifeq "<arg1>" '<arg2>'
??? ifeq '<arg1>' "<arg2>"
比較參數(shù)“arg1”和“arg2”的值是否相同。當(dāng)然,參數(shù)中我們還可以使用make的函數(shù)。如:
??? ifeq ($(strip $(foo)),)
??? <text-if-empty>
??? endif
這個(gè)示例中使用了“strip”函數(shù),如果這個(gè)函數(shù)的返回值是空(Empty),那么<text-if-empty>就生效。
第二個(gè)條件關(guān)鍵字是“ifneq”。語法是:
??? ifneq (<arg1>, <arg2>)
??? ifneq '<arg1>' '<arg2>'
??? ifneq "<arg1>" "<arg2>"
??? ifneq "<arg1>" '<arg2>'
??? ifneq '<arg1>' "<arg2>"
其比較參數(shù)“arg1”和“arg2”的值是否相同,如果不同,則為真。和“ifeq”類似。
第三個(gè)條件關(guān)鍵字是“ifdef”。語法是:
??? ifdef <variable-name>
如果變量<variable-name>的值非空,那到表達(dá)式為真。否則,表達(dá)式為假。當(dāng)然,<variable-name>同樣可以是一個(gè)函數(shù)的返回值。注意,ifdef只是測(cè)試一個(gè)變量是否有值,其并不會(huì)把變量擴(kuò)展到當(dāng)前位置。還是來看兩個(gè)例子:
??? 示例一:
??? bar =
??? foo = $(bar)
??? ifdef foo
??? frobozz = yes
??? else
??? frobozz = no
??? endif
??? 示例二:
??? foo =
??? ifdef foo
??? frobozz = yes
??? else
??? frobozz = no
??? endif
第一個(gè)例子中,“$(frobozz)”值是“yes”,第二個(gè)則是“no”。
第四個(gè)條件關(guān)鍵字是“ifndef”。其語法是:
??? ifndef <variable-name>
這個(gè)我就不多說了,和“ifdef”是相反的意思。
在<conditional-directive>這一行上,多余的空格是被允許的,但是不能以[Tab]鍵做為開始(不然就被認(rèn)為是命令)。而注釋符“#”同樣也是安全的。“else”和“endif”也一樣,只要不是以[Tab]鍵開始就行了。
特別注意的是,make是在讀取Makefile時(shí)就計(jì)算條件表達(dá)式的值,并根據(jù)條件表達(dá)式的值來選擇語句,所以,你最好不要把自動(dòng)化變量(如“$@”等)放入條件表達(dá)式中,因?yàn)樽詣?dòng)化變量是在運(yùn)行時(shí)才有的。
而且,為了避免混亂,make不允許把整個(gè)條件語句分成兩部分放在不同的文件中。
出處:http://blog.csdn.net/haoel/article/details/2893
總結(jié)
以上是生活随笔為你收集整理的跟我一起写 Makefile(八)的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 跟我一起写 Makefile(七)
- 下一篇: 跟我一起写 Makefile(九)