1. 程式人生 > >makefile筆記6 - makefile條件判斷

makefile筆記6 - makefile條件判斷

註釋符 表達 -i var else name 指定 text _for

使用條件判斷,可以讓 make 根據運行時的不同情況選擇不同的執行分支。條件表達式可以是比較變量的值,或是變量和常量的值。

一、示例

下面的例子,判斷\(\$\)(CC)變量是否“gcc”,如果是的話,則使用 GNU 函數編譯目標。

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

可見,在上面示例的這個規則中,目標“foo”可以根據變量“\(\$\)

(CC)”值來選取不同的函數庫來編譯程序。
我們可以從上面的示例中看到三個關鍵字: ifeq、 else 和 endif。 ifeq 的意思表示條件語句的開始,並指定一個條件表達式,表達式包含兩個參數,以逗號分隔,表達式以圓括號括起。 else 表示條件表達式為假的情況。 endif 表示一個條件語句的結束,任何一個條件表達式都應該以 endif 結束。
當我們的變量\(\$\)(CC)值是“gcc”時,目標 foo 的規則是:

foo: $(objects)
$(CC) -o foo $(objects) $(libs_for_gcc)

而當我們的變量$(CC)值不是“gcc”時(比如“cc”),目標 foo 的規則是:

foo: $(objects)
$(CC) -o foo $(objects) $(normal_libs)當然,我們還可以把上面的那個例子寫得更簡潔一些:
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)


二、語法

條件表達式的語法為:

<conditional-directive>
<text-if-true>
endif
以及:
<conditional-directive>
<text-if-true>
else
<text-if-false>
endif

其中<conditional-directive>表示條件關鍵字,如“ifeq”。這個關鍵字有四種。

1. 第一個是我們前面所見過的“ifeq”

ifeq (<arg1>, <arg2>)
ifeq ‘<arg1>‘ ‘<arg2>‘
ifeq "<arg1>" "<arg2>"
ifeq "<arg1>" ‘<arg2>‘
ifeq ‘<arg1>‘ "<arg2>"

比較參數“arg1”和“arg2”的值是否相同。當然,參數中我們還可以使用 make 的函數。如:

ifeq ($(strip $(foo)),)
<text-if-empty>
endif

這 個 示 例 中 使 用 了 “strip ” 函 數 , 如 果 這 個 函 數 的 返 回 值 是 空(Empty), 那 麽<text-if-empty>就生效。

2. 第二個條件關鍵字是“ifneq”。語法是:

ifneq (<arg1>, <arg2>)
ifneq ‘<arg1>‘ ‘<arg2>‘
ifneq "<arg1>" "<arg2>"
ifneq "<arg1>" ‘<arg2>‘
ifneq ‘<arg1>‘ "<arg2>"

其比較參數“arg1”和“arg2”的值是否相同,如果不同,則為真。和“ifeq”類似。

3. 第三個條件關鍵字是“ifdef”。語法是:

ifdef <variable-name>

如果變量<variable-name>的值非空,那到表達式為真。否則,表達式為假。當然,<variable-name>同樣可以是一個函數的返回值。註意, ifdef 只是測試一個變量是否有值,其並不會把變量擴展到當前位置。還是來看兩個例子:
示例一:

bar =
foo = $(bar)
ifdef foo
frobozz = yes
else
frobozz = no
endif

示例二:

foo =
ifdef foo
frobozz = yes
else
frobozz = no
endif

第一個例子中,“\(\$\)(frobozz)”值是“yes”,第二個則是“no”。

4. 第四個條件關鍵字是“ifndef”。其語法是:

ifndef <variable-name>

這個我就不多說了,和“ifdef”是相反的意思。
在<conditional-directive>這一行上,多余的空格是被允許的,但是不能以[Tab]鍵做為開始(不然就被認為是命令)。而註釋符“#”同樣也是安全的。“else”和“endif”也一樣,只要不是以[Tab]鍵開始就行了。
特別註意的是, make 是在讀取 Makefile 時就計算條件表達式的值,並根據條件表達式的值來選擇語句,所以,你最好不要把自動化變量(如“$@”等)放入條件表達式中,因為自動化變量是在運行時才有的。
而且,為了避免混亂, make 不允許把整個條件語句分成兩部分放在不同的文件中。



參考文獻
跟我一起寫makefile(絕大部分內容都是copy的這篇文章,因為這篇文章寫得已經很棒了)

makefile筆記6 - makefile條件判斷