• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Prayer

            在一般中尋求卓越
            posts - 1256, comments - 190, trackbacks - 0, articles - 0
              C++博客 :: 首頁(yè) :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理
            六、多目標(biāo)
            Makefile的規(guī)則中的目標(biāo)可以不止一個(gè),其支持多目標(biāo),有可能我們的多個(gè)目標(biāo)同時(shí)依賴于一個(gè)文件,并且其生成的命令大體類(lèi)似。于是我們就能把其合并起來(lái)。當(dāng)然,多個(gè)目標(biāo)的生成規(guī)則的執(zhí)行命令是同一個(gè),這可能會(huì)可我們帶來(lái)麻煩,不過(guò)好在我們的可以使用一個(gè)自動(dòng)化變量“$@”(關(guān)于自動(dòng)化變量,將在后面講述),這個(gè)變量表示著目前規(guī)則中所有的目標(biāo)的集合,這樣說(shuō)可能很抽象,還是看一個(gè)例子吧。

                bigoutput littleoutput : text.g
                        generate text.g -$(subst output,,$@) >; $@

                上述規(guī)則等價(jià)于:

                bigoutput : text.g
                        generate text.g -big >; bigoutput
                littleoutput : text.g
                        generate text.g -little >; littleoutput

                其中,-$(subst output,,$@)中的“$”表示執(zhí)行一個(gè)Makefile的函數(shù),函數(shù)名為subst,后面的為參數(shù)。關(guān)于函數(shù),將在后面講述。這里的這個(gè)函數(shù)是截取字符串的意思,“$@”表示目標(biāo)的集合,就像一個(gè)數(shù)組,“$@”依次取出目標(biāo),并執(zhí)于命令。


            七、靜態(tài)模式

            靜態(tài)模式可以更加容易地定義多目標(biāo)的規(guī)則,可以讓我們的規(guī)則變得更加的有彈性和靈活。我們還是先來(lái)看一下語(yǔ)法:

                <targets ...>;: <target-pattern>;: <prereq-patterns ...>;
                        <commands>;
                        ...


                targets定義了一系列的目標(biāo)文件,可以有通配符。是目標(biāo)的一個(gè)集合。

                target-parrtern是指明了targets的模式,也就是的目標(biāo)集模式。

                prereq-parrterns是目標(biāo)的依賴模式,它對(duì)target-parrtern形成的模式再進(jìn)行一次依賴目標(biāo)的定義。

            這樣描述這三個(gè)東西,可能還是沒(méi)有說(shuō)清楚,還是舉個(gè)例子來(lái)說(shuō)明一下吧。如果我們的<target-parrtern>;定義成“%.o”,意思是我們的<target>;集合中都是以“.o”結(jié)尾的,而如果我們的<prereq-parrterns>;定義成“%.c”,意思是對(duì)<target-parrtern>;所形成的目標(biāo)集進(jìn)行二次定義,其計(jì)算方法是,取<target-parrtern>;模式中的“%”(也就是去掉了[.o]這個(gè)結(jié)尾),并為其加上[.c]這個(gè)結(jié)尾,形成的新集合。

            所以,我們的“目標(biāo)模式”或是“依賴模式”中都應(yīng)該有“%”這個(gè)字符,如果你的文件名中有“%”那么你可以使用反斜杠“\”進(jìn)行轉(zhuǎn)義,來(lái)標(biāo)明真實(shí)的“%”字符。

            看一個(gè)例子:

                objects = foo.o bar.o

                all: $(objects)

                $(objects): %.o: %.c
                        $(CC) -c $(CFLAGS) $< -o $@


            上面的例子中,指明了我們的目標(biāo)從$object中獲取,“%.o”表明要所有以“.o”結(jié)尾的目標(biāo),也就是“foo.o bar.o”,也就是變量$object集合的模式,而依賴模式“%.c”則取模式“%.o”的“%”,也就是“foo bar”,并為其加下“.c”的后綴,于是,我們的依賴目標(biāo)就是“foo.c bar.c”。而命令中的“$<”和“$@”則是自動(dòng)化變量,“$<”表示所有的依賴目標(biāo)集(也就是“foo.c bar.c”),“$@”表示目標(biāo)集(也就是“foo.o bar.o”)。于是,上面的規(guī)則展開(kāi)后等價(jià)于下面的規(guī)則:

                foo.o : foo.c
                        $(CC) -c $(CFLAGS) foo.c -o foo.o
                bar.o : bar.c
                        $(CC) -c $(CFLAGS) bar.c -o bar.o

            試想,如果我們的“%.o”有幾百個(gè),那種我們只要用這種很簡(jiǎn)單的“靜態(tài)模式規(guī)則”就可以寫(xiě)完一堆規(guī)則,實(shí)在是太有效率了。“靜態(tài)模式規(guī)則”的用法很靈活,如果用得好,那會(huì)一個(gè)很強(qiáng)大的功能。再看一個(gè)例子:


                files = foo.elc bar.o lose.o

                $(filter %.o,$(files)): %.o: %.c
                        $(CC) -c $(CFLAGS) $< -o $@
                $(filter %.elc,$(files)): %.elc: %.el
                        emacs -f batch-byte-compile $<


            $(filter %.o,$(files))表示調(diào)用Makefile的filter函數(shù),過(guò)濾“$filter”集,只要其中模式為“%.o”的內(nèi)容。其的它內(nèi)容,我就不用多說(shuō)了吧。這個(gè)例字展示了Makefile中更大的彈性。


            八、自動(dòng)生成依賴性

            在Makefile中,我們的依賴關(guān)系可能會(huì)需要包含一系列的頭文件,比如,如果我們的main.c中有一句“#include "defs.h"”,那么我們的依賴關(guān)系應(yīng)該是:

                main.o : main.c defs.h

            但是,如果是一個(gè)比較大型的工程,你必需清楚哪些C文件包含了哪些頭文件,并且,你在加入或刪除頭文件時(shí),也需要小心地修改Makefile,這是一個(gè)很沒(méi)有維護(hù)性的工作。為了避免這種繁重而又容易出錯(cuò)的事情,我們可以使用C/C++編譯的一個(gè)功能。大多數(shù)的C/C++編譯器都支持一個(gè)“-M”的選項(xiàng),即自動(dòng)找尋源文件中包含的頭文件,并生成一個(gè)依賴關(guān)系。例如,如果我們執(zhí)行下面的命令:

                cc -M main.c

            其輸出是:

                main.o : main.c defs.h

            于是由編譯器自動(dòng)生成的依賴關(guān)系,這樣一來(lái),你就不必再手動(dòng)書(shū)寫(xiě)若干文件的依賴關(guān)系,而由編譯器自動(dòng)生成了。需要提醒一句的是,如果你使用GNU的C/C++編譯器,你得用“-MM”參數(shù),不然,“-M”參數(shù)會(huì)把一些標(biāo)準(zhǔn)庫(kù)的頭文件也包含進(jìn)來(lái)。

                gcc -M main.c的輸出是:

                main.o: main.c defs.h /usr/include/stdio.h /usr/include/features.h \
                     /usr/include/sys/cdefs.h /usr/include/gnu/stubs.h \
                     /usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stddef.h \
                     /usr/include/bits/types.h /usr/include/bits/pthreadtypes.h \
                     /usr/include/bits/sched.h /usr/include/libio.h \
                     /usr/include/_G_config.h /usr/include/wchar.h \
                     /usr/include/bits/wchar.h /usr/include/gconv.h \
                     /usr/lib/gcc-lib/i486-suse-linux/2.95.3/include/stdarg.h \
                     /usr/include/bits/stdio_lim.h


                gcc -MM main.c的輸出則是:

                main.o: main.c defs.h

            那么,編譯器的這個(gè)功能如何與我們的Makefile聯(lián)系在一起呢。因?yàn)檫@樣一來(lái),我們的Makefile也要根據(jù)這些源文件重新生成,讓Makefile自已依賴于源文件?這個(gè)功能并不現(xiàn)實(shí),不過(guò)我們可以有其它手段來(lái)迂回地實(shí)現(xiàn)這一功能。GNU組織建議把編譯器為每一個(gè)源文件的自動(dòng)生成的依賴關(guān)系放到一個(gè)文件中,為每一個(gè)“name.c”的文件都生成一個(gè)“name.d”的Makefile文件,[.d]文件中就存放對(duì)應(yīng)[.c]文件的依賴關(guān)系。

            于是,我們可以寫(xiě)出[.c]文件和[.d]文件的依賴關(guān)系,并讓make自動(dòng)更新或自成[.d]文件,并把其包含在我們的主Makefile中,這樣,我們就可以自動(dòng)化地生成每個(gè)文件的依賴關(guān)系了。

            這里,我們給出了一個(gè)模式規(guī)則來(lái)產(chǎn)生[.d]文件:

                %.d: %.c
                        @set -e; rm -f $@; \
                         $(CC) -M $(CPPFLAGS) $< >; $@.$$$$; \
                         sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ >; $@; \
                         rm -f $@.$$$$


            這個(gè)規(guī)則的意思是,所有的[.d]文件依賴于[.c]文件,“rm -f $@”的意思是刪除所有的目標(biāo),也就是[.d]文件,第二行的意思是,為每個(gè)依賴文件“$<”,也就是[.c]文件生成依賴文件,“$@”表示模式“%.d”文件,如果有一個(gè)C文件是name.c,那么“%”就是“name”,“$$$$”意為一個(gè)隨機(jī)編號(hào),第二行生成的文件有可能是“name.d.12345”,第三行使用sed命令做了一個(gè)替換,關(guān)于sed命令的用法請(qǐng)參看相關(guān)的使用文檔。第四行就是刪除臨時(shí)文件。

            總而言之,這個(gè)模式要做的事就是在編譯器生成的依賴關(guān)系中加入[.d]文件的依賴,即把依賴關(guān)系:

                main.o : main.c defs.h

            轉(zhuǎn)成:

                main.o main.d : main.c defs.h

            于是,我們的[.d]文件也會(huì)自動(dòng)更新了,并會(huì)自動(dòng)生成了,當(dāng)然,你還可以在這個(gè)[.d]文件中加入的不只是依賴關(guān)系,包括生成的命令也可一并加入,讓每個(gè)[.d]文件都包含一個(gè)完賴的規(guī)則。一旦我們完成這個(gè)工作,接下來(lái),我們就要把這些自動(dòng)生成的規(guī)則放進(jìn)我們的主Makefile中。我們可以使用Makefile的“include”命令,來(lái)引入別的Makefile文件(前面講過(guò)),例如:

                sources = foo.c bar.c

                include $(sources:.c=.d)

            上述語(yǔ)句中的“$(sources:.c=.d)”中的“.c=.d”的意思是做一個(gè)替換,把變量$(sources)所有[.c]的字串都替換成[.d],關(guān)于這個(gè)“替換”的內(nèi)容,在后面我會(huì)有更為詳細(xì)的講述。當(dāng)然,你得注意次序,因?yàn)閕nclude是按次來(lái)載入文件,最先載入的[.d]文件中的目標(biāo)會(huì)成為默認(rèn)目標(biāo)。

            欧美噜噜久久久XXX| 色婷婷综合久久久久中文字幕 | 久久精品一区二区三区中文字幕| 久久天天躁狠狠躁夜夜avapp | 日本精品一区二区久久久| 久久国产成人午夜aⅴ影院| 亚洲国产成人久久综合一| 色成年激情久久综合| 精品久久久久久久久久中文字幕 | 久久精品成人免费网站| 色综合久久综合网观看| 精品无码久久久久久久久久| 99久久免费只有精品国产| 精品一久久香蕉国产线看播放| 久久国产影院| 亚洲AV日韩AV天堂久久| 国产亚洲精品自在久久| 久久久久久极精品久久久| 久久久久久久女国产乱让韩| 久久久久人妻一区二区三区vr | 久久夜色精品国产亚洲av| 日韩久久无码免费毛片软件| 漂亮人妻被黑人久久精品| 久久久久一区二区三区| 亚洲国产天堂久久综合| 久久男人Av资源网站无码软件| 99久久精品国产一区二区| 色老头网站久久网| 99久久精品午夜一区二区| 国内精品欧美久久精品| 中文字幕无码免费久久| 成人精品一区二区久久久| 国产69精品久久久久久人妻精品 | 国产成人综合久久综合| 无码国内精品久久人妻麻豆按摩| 欧美熟妇另类久久久久久不卡| 久久99精品久久久久久不卡| 亚洲国产精品久久电影欧美| 久久最新免费视频| 精品人妻伦一二三区久久| 99久久99久久精品免费看蜜桃|