1. 程式人生 > >跟我一起寫 Makefile - 6. 使用條件判斷

跟我一起寫 Makefile - 6. 使用條件判斷

跟我一起寫 Makefile - 6. 使用條件判斷

http://wiki.ubuntu.org.cn

6. 使用條件判斷

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

6.1 示例

下面的例子,判斷 $(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)

6.2 語法

條件表示式的語法為:

<conditional-directive>
<text-if-true>
endif

以及:

<conditional-directive>
<text-if-true>
else
<text-if-false>
endif

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

第一個是我們前面所見過的 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>; 就生效。

第二個條件關鍵字是 ifneq。語法是:

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

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

第三個條件關鍵字是 ifdef。語法是:

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

第四個條件關鍵字是 ifndef。其語法是:

ifndef <variable-name>

這個我就不多說了,和 ifdef 是相反的意思。

在 <conditional-directive> 這一行上,多餘的空格是被允許的,但是不能以 Tab 鍵作為開始 (不然就被認為是命令)。而註釋符 # 同樣也是安全的。else 和 endif 也一樣,只要不是以 Tab 鍵開始就行了。

特別注意的是,make 是在讀取 Makefile 時就計算條件表示式的值,並根據條件表示式的值來選擇語句,所以你最好不要把自動化變數 (如 [email protected] 等) 放入條件表示式中,因為自動化變數是在執行時才有的。

為了避免混亂,make 不允許把整個條件語句分成兩部分放在不同的檔案中。

Wordbook

GNU Compiler Collection,GCC:GNU 編譯器套件
GNU C Compiler,GCC:GNU C 編譯器
Microsoft Visual C++,MSVC
Unix / UNIX /ˈjuːnɪks/