天天看點

跟我一起寫 Makefile(九)

使用函數

————

在Makefile中可以使用函數來處理變量,進而讓我們的指令或是規則更為的靈活和具有智能。make所支援的函數也不算很多,不過已經足夠我們的操作了。函數調用後,函數的傳回值可以當做變量來使用。

一、函數的調用文法

函數調用,很像變量的使用,也是以“$”來辨別的,其文法如下:

    $(<function> <arguments>)

或是

    ${<function> <arguments>}

這裡,<function>就是函數名,make支援的函數不多。<arguments>是函數的參數,參數間以逗号“,”分隔,而函數名和參數之間以“空格”分隔。函數調用以“$”開頭,以圓括号或花括号把函數名和參數括起。感覺很像一個變量,是不是?函數中的參數可以使用變量,為了風格的統一,函數和變量的括号最好一樣,如使用“$(subst a,b,$(x))”這樣的形式,而不是“$(subst a,b,${x})”的形式。因為統一會更清楚,也會減少一些不必要的麻煩。

還是來看一個示例:

    comma:= ,

    empty:=

    space:= $(empty) $(empty)

    foo:= a b c

    bar:= $(subst $(space),$(comma),$(foo))

在這個示例中,$(comma)的值是一個逗号。$(space)使用了$(empty)定義了一個空格,$(foo)的值是“a b c”,$(bar)的定義用,調用了函數“subst”,這是一個替換函數,這個函數有三個參數,第一個參數是被替換字串,第二個參數是替換字串,第三個參數是替換操作作用的字串。這個函數也就是把$(foo)中的空格替換成逗号,是以$(bar)的值是“a,b,c”。

二、字元串處理函數

$(subst <from>,<to>,<text>)

    名稱:字元串替換函數——subst。

    功能:把字串<text>中的<from>字元串替換成<to>。

    傳回:函數傳回被替換過後的字元串。

    示例:

        $(subst ee,EE,feet on the street),

        把“feet on the street”中的“ee”替換成“EE”,傳回結果是“fEEt on the strEEt”。

$(patsubst <pattern>,<replacement>,<text>)

    名稱:模式字元串替換函數——patsubst。

    功能:查找<text>中的單詞(單詞以“空格”、“Tab”或“回車”“換行”分隔)是否符合模式<pattern>,如果比對的話,則以<replacement>替換。這裡,<pattern>可以包括通配符“%”,表示任意長度的字串。如果<replacement>中也包含“%”,那麼,<replacement>中的這個“%”将是<pattern>中的那個“%”所代表的字串。(可以用“\”來轉義,以“\%”來表示真實含義的“%”字元)

        $(patsubst %.c,%.o,x.c.c bar.c)

        把字串“x.c.c bar.c”符合模式[%.c]的單詞替換成[%.o],傳回結果是“x.c.o bar.o”

    備注:

        這和我們前面“變量章節”說過的相關知識有點相似。如:

        “$(var:<pattern>=<replacement>)”

         相當于

        “$(patsubst <pattern>,<replacement>,$(var))”,

         而“$(var: <suffix>=<replacement>)”

         則相當于

         “$(patsubst %<suffix>,%<replacement>,$(var))”。

         例如有:objects = foo.o bar.o baz.o,

         那麼,“$(objects:.o=.c)”和“$(patsubst %.o,%.c,$(objects))”是一樣的。

$(strip <string>)

    名稱:去空格函數——strip。

    功能:去掉<string>字串中開頭和結尾的空字元。

    傳回:傳回被去掉空格的字元串值。

        $(strip a b c )

        把字串“a b c ”去到開頭和結尾的空格,結果是“a b c”。

$(findstring <find>,<in>)

    名稱:查找字元串函數——findstring。

    功能:在字串<in>中查找<find>字串。

    傳回:如果找到,那麼傳回<find>,否則傳回空字元串。

        $(findstring a,a b c)

        $(findstring a,b c)

        第一個函數傳回“a”字元串,第二個傳回“”字元串(空字元串)

$(filter <pattern...>,<text>)

    名稱:過濾函數——filter。

    功能:以<pattern>模式過濾<text>字元串中的單詞,保留符合模式<pattern>的單詞。可以有多個模式。

    傳回:傳回符合模式<pattern>的字串。

        sources := foo.c bar.c baz.s ugh.h

        foo: $(sources)

                cc $(filter %.c %.s,$(sources)) -o foo

        $(filter %.c %.s,$(sources))傳回的值是“foo.c bar.c baz.s”。

$(filter-out <pattern...>,<text>)

    名稱:反過濾函數——filter-out。

    功能:以<pattern>模式過濾<text>字元串中的單詞,去除符合模式<pattern>的單詞。可以有多個模式。

    傳回:傳回不符合模式<pattern>的字串。

        objects=main1.o foo.o main2.o bar.o

        mains=main1.o main2.o

        $(filter-out $(mains),$(objects)) 傳回值是“foo.o bar.o”。

$(sort <list>)

    名稱:排序函數——sort。

    功能:給字元串<list>中的單詞排序(升序)。

    傳回:傳回排序後的字元串。

    示例:$(sort foo bar lose)傳回“bar foo lose” 。

    備注:sort函數會去掉<list>中相同的單詞。

$(word <n>,<text>)

    名稱:取單詞函數——word。

    功能:取字元串<text>中第<n>個單詞。(從一開始)

    傳回:傳回字元串<text>中第<n>個單詞。如果<n>比<text>中的單詞數要大,那麼傳回空字元串。

    示例:$(word 2, foo bar baz)傳回值是“bar”。

$(wordlist <s>,<e>,<text>) 

    名稱:取單詞串函數——wordlist。

    功能:從字元串<text>中取從<s>開始到<e>的單詞串。<s>和<e>是一個數字。

    傳回:傳回字元串<text>中從<s>到<e>的單詞字串。如果<s>比<text>中的單詞數要大,那麼傳回空字元串。如果<e>大于<text>的單詞數,那麼傳回從<s>開始,到<text>結束的單詞串。

    示例: $(wordlist 2, 3, foo bar baz)傳回值是“bar baz”。

$(words <text>)

    名稱:單詞個數統計函數——words。

    功能:統計<text>中字元串中的單詞個數。

    傳回:傳回<text>中的單詞數。

    示例:$(words, foo bar baz)傳回值是“3”。

    備注:如果我們要取<text>中最後的一個單詞,我們可以這樣:$(word $(words <text>),<text>)。

$(firstword <text>)

    名稱:首單詞函數——firstword。

    功能:取字元串<text>中的第一個單詞。

    傳回:傳回字元串<text>的第一個單詞。

    示例:$(firstword foo bar)傳回值是“foo”。

    備注:這個函數可以用word函數來實作:$(word 1,<text>)。

以上,是所有的字元串操作函數,如果搭配混合使用,可以完成比較複雜的功能。這裡,舉一個現實中應用的例子。我們知道,make使用“VPATH”變量來指定“依賴檔案”的搜尋路徑。于是,我們可以利用這個搜尋路徑來指定編譯器對頭檔案的搜尋路徑參數CFLAGS,如:

    override CFLAGS += $(patsubst %,-I%,$(subst :, ,$(VPATH)))

    如果我們的“$(VPATH)”值是“src:../headers”,那麼“$(patsubst %,-I%,$(subst :, ,$(VPATH)))”将傳回“-Isrc -I../headers”,這正是cc或gcc搜尋頭檔案路徑的參數。

三、檔案名操作函數

下面我們要介紹的函數主要是處理檔案名的。每個函數的參數字元串都會被當做一個或是一系列的檔案名來對待。

$(dir <names...>)

    名稱:取目錄函數——dir。

    功能:從檔案名序列<names>中取出目錄部分。目錄部分是指最後一個反斜杠(“/”)之前的部分。如果沒有反斜杠,那麼傳回“./”。

    傳回:傳回檔案名序列<names>的目錄部分。

    示例: $(dir src/foo.c hacks)傳回值是“src/ ./”。

$(notdir <names...>)

    名稱:取檔案函數——notdir。

    功能:從檔案名序列<names>中取出非目錄部分。非目錄部分是指最後一個反斜杠(“/”)之後的部分。

    傳回:傳回檔案名序列<names>的非目錄部分。

    示例: $(notdir src/foo.c hacks)傳回值是“foo.c hacks”。

$(suffix <names...>) 

    名稱:取字尾函數——suffix。

    功能:從檔案名序列<names>中取出各個檔案名的字尾。

    傳回:傳回檔案名序列<names>的字尾序列,如果檔案沒有字尾,則傳回空字串。

    示例:$(suffix src/foo.c src-1.0/bar.c hacks)傳回值是“.c .c”。

$(basename <names...>)

    名稱:取字首函數——basename。

    功能:從檔案名序列<names>中取出各個檔案名的字首部分。

    傳回:傳回檔案名序列<names>的字首序列,如果檔案沒有字首,則傳回空字串。

    示例:$(basename src/foo.c src-1.0/bar.c hacks)傳回值是“src/foo src-1.0/bar hacks”。

$(addsuffix <suffix>,<names...>)

    名稱:加字尾函數——addsuffix。

    功能:把字尾<suffix>加到<names>中的每個單詞後面。

    傳回:傳回加過字尾的檔案名序列。

    示例:$(addsuffix .c,foo bar)傳回值是“foo.c bar.c”。

$(addprefix <prefix>,<names...>)

    名稱:加字首函數——addprefix。

    功能:把字首<prefix>加到<names>中的每個單詞後面。

    傳回:傳回加過字首的檔案名序列。

    示例:$(addprefix src/,foo bar)傳回值是“src/foo src/bar”。

$(join <list1>,<list2>)

    名稱:連接配接函數——join。

    功能:把<list2>中的單詞對應地加到<list1>的單詞後面。如果<list1>的單詞個數要比<list2>的多,那麼,<list1>中的多出來的單詞将保持原樣。如果<list2>的單詞個數要比<list1>多,那麼,<list2>多出來的單詞将被複制到<list2>中。

    傳回:傳回連接配接過後的字元串。

    示例:$(join aaa bbb , 111 222 333)傳回值是“aaa111 bbb222 333”。

本文轉自 haoel 51CTO部落格,原文連結:http://blog.51cto.com/haoel/124639,如需轉載請自行聯系原作者

繼續閱讀