版權(quán)說(shuō)明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡(jiǎn)介
第一部分、概 第二部分、關(guān)于程序的編譯和...............................................................................................第三部分、Makefile介 二、一個(gè)示 六、另類風(fēng)格的 七、清空目標(biāo)文件的規(guī) 第四部分、Makefile總 三、其它的 四、環(huán)境變量 第五部分、書(shū)寫規(guī) 一、規(guī)則舉 二、規(guī)則的語(yǔ) 三、在規(guī)則中使用通配 四、文件搜 五、偽目 六、多目 七、靜態(tài)模 八、自動(dòng)生成依賴 第六部分書(shū)寫命 一、顯示命 二、命令執(zhí) 三、命令出 四、嵌套執(zhí)行 五、定義命令 第七部分使用變 一、變量的基 二、變量中的變 三、變量高級(jí)用 四、追加變量 五、override指示 六、多行變 七、環(huán)境變 八、目標(biāo)變 九、模式變 第八部分使用條件判 一、示 二、語(yǔ) 第九部分使用函 一、函數(shù)的調(diào)用語(yǔ) 二、字符串處理函 三、文件名操作函 四、foreach函 五、if函 第十部分make的運(yùn) 二、指定 三、指定目 四、檢查規(guī) 第十一部分隱含規(guī) 一、使用隱含規(guī) 二、隱含規(guī)則一 10、LexC程序時(shí)的隱含規(guī) 三、隱含規(guī)則使用的變 四、隱含規(guī)則 五、定義模式規(guī) 七、隱含規(guī)則搜索算 第十二部分使用make更新函數(shù)庫(kù)文 一、函數(shù)庫(kù)文件的成 二、函數(shù)庫(kù)成員的隱含規(guī) 三、函數(shù)庫(kù)文件的后綴規(guī) 四、注意事 第十三部分后 什么是makefile?或許很多Winodws的程序員都不知道這個(gè)東西,因?yàn)槟切¦indows還是要懂。這就好像現(xiàn)在有這么多的HTML的編輯器,但如果你想成為一個(gè)專業(yè),你還會(huì)不會(huì)寫makefile,從一個(gè)側(cè)面說(shuō)明了一個(gè)人是否具備完成大型工程的能力。因?yàn)?,makefile關(guān)系到了整個(gè)工程的編譯規(guī)則。一個(gè)工程中的源文件不計(jì)數(shù),其按類型、功能、 哪些文件需要后編譯,哪些文件需要重新編譯,甚至于進(jìn)行更復(fù)雜的功能操作,因?yàn)樘幘褪恰白詣?dòng)化編譯”,一旦寫好,只需要一個(gè)make命令,整個(gè)工程完全自動(dòng)編譯,Linux下GNU的make??梢?jiàn),makefile都成為了一種在工程方面的編譯方法?,F(xiàn)在講述如何寫makefile的文章比較少,這是寫這篇文章的原因。當(dāng)然,不同產(chǎn)make(POSIX.2UNIX下的GCC和CC。Windows.obj文件,UNIX.o文(compile(link明,而定義應(yīng)該放在C/C++文件中,只要所有的語(yǔ)法正確,編譯器就可以編譯出中間目標(biāo)文件。一般來(lái)說(shuō),每個(gè)源文件都應(yīng)該對(duì)應(yīng)于一個(gè)中間目標(biāo)文件(OOBJ。鏈文件)來(lái)我們的應(yīng)用程序。器并不管函數(shù)所在的源文件,函數(shù)的中間目標(biāo)文件File下,是ArchiveFile,也就是.a文件。ErrorObjectFile.第三部分、MakefileMakefile如果這個(gè)工程的頭文件被改變了,那么我們需要編譯了這幾個(gè)頭文件的C文件,maketarget...:prerequisites...(Labelprerequisites就是,要生成那個(gè)target所需要的文件或是目標(biāo)。這是一個(gè)文件的依賴關(guān)系,也就是說(shuō),target這一個(gè)或多個(gè)的目標(biāo)文件依賴于中如果有一個(gè)以上的文件比target文件要新的話,command所定義令就會(huì)被執(zhí)行。這就是Makefile的規(guī)則。也就是Makefile中最的內(nèi)容。說(shuō)到底,Makefile的那三個(gè)規(guī)則,我們的Makefile應(yīng)該是下面的這個(gè)樣子的。edit:main.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.occ-oeditmain.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.omain.o:main.cdefs.hcc-cmain.ckbd.o:kbd.cdefs.hcommand.hcc-ckbd.ccommand.o:command.cdefs.hcommand.hcc-ccommand.cdisy.o:disy.cdefs.hbuffer.hcc-cdisy.cinsert.o:insert.cdefs.hbuffer.hcc-cinsert.csearch.o:search.cdefs.hbuffer.hcc-csearch.cfiles.o:files.cdefs.hbuffer.hcommand.hcc-cfiles.cutils.o:utils.cdefs.hcc-cutils.ccleanrmeditmain.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.o存在文件為“Makefile”或“makefile”的文件中,然后在該下直接輸入命令“make”edit。如果要?jiǎng)h除執(zhí)行文件和所有的中間目標(biāo)文件,那么,只要簡(jiǎn)單地執(zhí)行一下“makeclean”就可以了。(*.o依賴文件(prerequisites).c.h文件。每一個(gè).o文件都.oedit義令。不會(huì)自動(dòng)執(zhí)行其后所定義令。要執(zhí)行其后令,就要在make命令后明顯得這個(gè)編譯無(wú)關(guān)令,比如程序的打包,程序的備份,等等。1、make會(huì)在當(dāng) (target這個(gè)文件新,那么,他就會(huì)執(zhí)行后面所定義令來(lái)生成edit這個(gè)文件。賴性,如果找到則再根據(jù)那一個(gè)規(guī)則生成.o(這有點(diǎn)像一個(gè)堆棧的過(guò)程)件生命make的終極任務(wù),也就是執(zhí)行文件edit了。make就會(huì)直接退出,并報(bào)錯(cuò),而對(duì)于所定義令的錯(cuò)誤,或是編譯不成功,make根本不么它后面所定義令將不會(huì)被自動(dòng)執(zhí)行,不過(guò),我們可以顯示要make執(zhí)行。即命令edit要新,所以edit也會(huì)被重新了(詳見(jiàn)edit目標(biāo)文件后定義令。而如果我們改變了“command.h”,那么,、mand.ofiles.o并且,edit會(huì)被重。edit:main.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.occ-oeditmain.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.o件,那么我們需要在兩個(gè)地方加(clean。當(dāng)然,我比如,我們一個(gè)變量,叫objects,OBJECTS,objs,OBJS,obj,或是OBJ,反正不管什么啦,只要能夠表示objmakefile一開(kāi)始就這樣定義:objects=main.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.o變量了,于是我們的改良版makefile就變成下面這個(gè)樣子:objects=main.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.oedit:cc-oedit$(objects)main.o:main.cdefs.hcc-cmain.ckbd.o:kbd.cdefs.hcommand.hcc-ckbd.ccommand.o:command.cdefs.hcommand.hcc-ccommand.cdisy.o:disy.cdefs.hbuffer.hcc-cdisy.cinsert.o:insert.cdefs.hbuffer.hcc-cinsert.csearch.o:search.cdefs.hbuffer.hcc-csearch.cfiles.o:files.cdefs.hbuffer.hcommand.hcc-cfiles.cutils.o:utils.cdefs.hcc-cutils.ccleanrmedit關(guān)于變量的話題,我會(huì)在后續(xù)給你一一道來(lái)。GNU的make很強(qiáng)大,它可以自動(dòng)推導(dǎo)文件以及文件依賴關(guān)系后面令,于是我們就沒(méi)必make[.o]文件,它就會(huì)自動(dòng)的把[.cmake找到一個(gè)whatever.o,那么whatever.c,就會(huì)是whatever.o的依賴文件。并且cc-c的makefile又出爐了。objects=main.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.oedit:cc-oeditmain.o:kbd.o:defs.hcommand.hcommand.o:defs.hcommand.hdisy.o:defs.hbuffer.hinsert.o:defs.hbuffer.hsearch.o:defs.hbuffer.hfiles.o:defs.hbuffer.hcommand.hutils.o:defs.h.PHONY:cleanclean:rmedit誰(shuí)叫它提供了自動(dòng)推導(dǎo)命令和文件的功能呢?來(lái)看看風(fēng)格的makefile吧。objects=main.okbd.ocommand.odisy.o\insert.osearch.ofiles.outils.oedit:cc-oedit$(objects):kbd.ocommand.ofiles.o:disy.oinsert.osearch.ofiles.o:.PHONY:cleanclean:rmedit楚,二是如果文件一多,要加入幾個(gè)新的.o文件,那就理不清楚了。rmedit.PHONY:cleanclean:-rmedit文的規(guī)矩是——“clean從來(lái)都是放在文件的最后”。第四部分、Makefile出,要生成的文件,文件的依賴文件,生成令。寫Makefile,這是由make所支持的。的宏,當(dāng)Makefile被執(zhí)行時(shí),其中的變量都會(huì)被擴(kuò)展到相應(yīng)的位置上。Makefile中只有行注釋,和UNIX的S一樣,其注釋是用“#”字符,這個(gè)就 默認(rèn)的情況下,make命令會(huì)在當(dāng)前 下按順序找尋文件名為“GNUmakefile”、寫的“makefile”文件名敏感,但是基本上來(lái)說(shuō),大多數(shù)的make都支持“makefile”和當(dāng)然,你可以使用別的文件名來(lái)書(shū)寫Makefile“Make.Linux”“Make.Solaris”,“Make.AIX”等,如果要指定特定的Makefile,make“-f”和“--file”參數(shù),如:make-fMake.Linuxmake--fileMake.AIX。MakefileincludeMakefileC#include,被包含的文件會(huì)原模原樣的放在當(dāng)前文件的包含位置。include的語(yǔ)法是:include<filename>前面可以有一些空字符,但是絕不能是[Tab]鍵開(kāi)始。include<filename>可以用一個(gè)或Makefile:a.mk、b.mk、c.mk,還有一個(gè)文件叫foo.make,以及一個(gè)變量$(bar),其包含了e.mk和f.mk,那么,下面的語(yǔ)句:includefoo.make*.mkincludefoo.makea.mkb.mkc.mke.mkmake會(huì)在當(dāng)前 所指定的下去尋找。2、如果<prefix>/include(一般是:/usr/local/bin/usr/include)存在的話,不理那些無(wú)法的文件,而繼續(xù)執(zhí)行,你可以在include前加一個(gè)減號(hào)“-”。includesinclude的文件發(fā)現(xiàn)錯(cuò)誤,make也會(huì)不理。是為了告訴大家,也許有時(shí)候你的Makefile出現(xiàn)了怪事,那么你當(dāng)前環(huán)境中有沒(méi)makeMakefilefoo.ofoo.cdefs.hfoocc-c-g看到這個(gè)例子,各位應(yīng)該不是很陌生了,前面也已,foo.o是我們的目標(biāo),foo.c文件日期要比f(wàn)oo.o文件日期要新,或是foo.o不存在,那么依賴關(guān)系發(fā)生。targets:prerequisitestargets:prerequisites;command沒(méi)有限制。規(guī)則告訴make兩件事,文件的依賴關(guān)系和如何成成目標(biāo)文件。如果我們想定義一系列比較類似的文件,我們很自然地就想起使用通配符。maketest。而“~hchen/testhchen的宿主下的test。(這些都Unix,make)WindowsMS-DOS下,用戶沒(méi)有宿主,那么波浪號(hào)所指的則根據(jù)環(huán)境變量“HOME”而定。rm-fprint:*.clpr-p$?printc]文objects=開(kāi),也就是讓objects的值是所有[.o]的文件名的集合,那么,你可以這樣:objects:=$(wildcard放在不同的make徑,但最好的方法是把一個(gè)路徑告訴make,讓make在自動(dòng)去找。make只會(huì)在當(dāng)前的中去找尋依賴文件和目標(biāo)文件。如果定義了這個(gè)變量,那么,make就會(huì)在當(dāng)當(dāng)前找不到的情況下,到所指定的中去找尋文件了。VPATH=上面的的定義指定兩個(gè),“src”和“../headers”,make索。由“冒號(hào)”分隔。(當(dāng)然,當(dāng)前是最高優(yōu)先搜索的地方)make“vpath”關(guān)鍵字(注意,它是全小寫它更為靈活。它可以指定不同的文件在不同的搜索中。這是一個(gè)很靈活的功能。它的使1、vpathpattern 2、vpath 3、 例如,“%.h”表示所有以“.h”結(jié)尾的文件。<pattern>指定了要搜索的文件集,而 vpath%.h該語(yǔ)句表示,要求make在“../headers” vpath%.cfoovpath%blishvpath%.c vpath%.cfoo:barvpath%blish而上面的語(yǔ)句則表示“.c”結(jié)尾的文件,先在“foo”,然后是“bar”,最后才是“blish”。rm*.o(以“makeclean”來(lái)使用該目.PHONY:rm*.o都寫在一個(gè)Makefile中,那么你可以使用“偽目標(biāo)”這個(gè)特性:all:prog1prog2.PHONY:prog1:prog1.occ-oprog1prog1.oprog2:cc-oprog2prog3:prog3.osort.occ-oprog3prog3.osort.o到了我們一口氣生成多個(gè)目標(biāo)的目的。“.PHONY:all”了“all”這個(gè)目標(biāo)為“偽目.PHONY:clelcleanobjcleandiffclel:cleanobjcleandiffrmcleanobjrmcleandiffrm“makeclean”將清除所有要被清除的文件?!癱leanobj”和“cleandiff”這兩個(gè)偽目標(biāo)有點(diǎn)像“子程序”的意思。我們可以輸入“makeclel”和“makecleanobj”和“makecleandiff”命令來(lái)達(dá)到清除不同種類文件的目的。Makefilebigoutputlittleoutput:generatetext.g-$(substoutput,,$@)>bigoutput:generatetext.g-big>bigoutputlittleoutput:text.ggeneratetext.g-little>活<targets...>:<target-pattern>:<prereq-patternsobjects=foo.obar.oall:$(objects)$(objects):%.o:$(CC)-c$(CFLAGS)$<-o上面的例子中,指明了我們的目標(biāo)從$object中獲取,“%.o”表明要所有以“.o”結(jié)尾的目標(biāo),也就是“foo.obar.o”,也就是變量$objectc”則取模式“%.o”的“%”,也就是“foobar”,并為其加下“.c”的后綴,于是,我們的依賴目標(biāo)就是“foo.cbar.c”。而命令中的“$<”和“$@”則是自動(dòng)化變量,“$<”表示bar.c”$@bar.o”foo.o:$(CC)-c$(CFLAGS)foo.c-ofoo.obar.o:bar.c$(CC)-c$(CFLAGS)bar.c-ofiles=foo.elcbar.o$(filter%.o,$(files)):%.o:$(CC)-c$(CFLAGS)$<-o$(filter%.elc,$(files)):%.elc:%.elemacs-f pile$<有一句“#include"defs.h"”,那么我們的依賴關(guān)系應(yīng)該是:main.o:main.c你在加入或刪除頭文件時(shí),也需要地修改Makefile,這是一個(gè)很沒(méi)有性的工作。為了避免這種繁重而又容易出錯(cuò)的事情,我們可以使用C/C++編譯的一個(gè)功能。大多數(shù)的賴關(guān)系。例如,如果我們執(zhí)行下面令:cc-Mmain.o:main.cgccMmain.cmain.o:main.cdefs.h/usr/include/stdio.h/usr/include/features.h/usr/include/sys/cdefs.h/usr/include/gnu/stubs.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.hmain.o:main.c文件的自動(dòng)生成的依賴關(guān)系放到一個(gè)文件中,為每一個(gè)“name.c”的文件都生成一個(gè)Makefile%.d:@set-e;rm-f$@;$(CC)-M$(CPPFLAGS)$<>$@.$$$$;sed's,\($*\)\.o[:]*,\1.o$@:,g'<$@.$$$$>$@;\rm-f$@.$$$$依賴文件,“$@”表示模式“%.d”文件,如果有一個(gè)C文件是name.c,那么“%”就是main.o:main.cmain.omain.d:main.csources=foo.cbar.cinclude$(sources:.c=.d)上述語(yǔ)句中的“$(sources:.c=.d)”中的“.c=.d”的意思是做一個(gè)替換,把變量每條規(guī)則中令和操作系統(tǒng)S令行是一致的。make會(huì)一按順序一條一條的make會(huì)認(rèn)為其是一個(gè)空命令。我們?cè)赨NIX下可能會(huì)使用不同的S,但是make令默認(rèn)是被“/bin/sh”——UNIX的標(biāo)準(zhǔn)S解釋執(zhí)行的。除非你特別指定一個(gè)其它的S。Makefile中,“#”是注釋符,很像C/C++中的“//”,其后的本行字符都被注釋。 echoXXX但不會(huì)執(zhí)行命令,這個(gè)功能很有利于我們調(diào)試我們的Makefile,看看我們書(shū)寫令是執(zhí)當(dāng)依賴目標(biāo)新于目標(biāo)時(shí),也就是當(dāng)規(guī)則的目標(biāo)需要被更新時(shí),makecd/home/hchen;當(dāng)我們執(zhí)行“makeexec”時(shí),第一個(gè)例子中的cd沒(méi)有作用,pwd會(huì)打印出當(dāng)前的 ,而第二個(gè)例子中,cd就起作用了,pwd會(huì)打印出“/home/hchen”。make一般是使用環(huán)境變量S中所定義的系統(tǒng)S來(lái)執(zhí)行命令,默認(rèn)情況下使用UNIX的環(huán)境變量,當(dāng)然你也可以指定。如果你指定了UNIX風(fēng)格的 形式,首先,make會(huì)在S 不存在,那么mkdir就成功執(zhí)行,萬(wàn)事大吉,如果 我們之所以使用mkdir的意思就是一定要有這樣的一個(gè) -rm-fmakei”或是“--ignore-errors”參數(shù),那么,是,如果某規(guī)則中令出錯(cuò)了,那么就終目該規(guī)則的執(zhí)行,但繼續(xù)執(zhí)行其它規(guī)則。 加地簡(jiǎn)潔,而不至于把所有的東西全部寫在一個(gè)Makefile中,這樣會(huì)很難我們的 叫subdir,這個(gè) 錄下文件的編譯規(guī)則。那么我們總控的Makefile可以這樣書(shū)寫:cdsubdir&&$(MAKE)-C較利于。這兩個(gè)例子的意思都是先進(jìn)入“subdir”,然后執(zhí)行make命令。export<variableunexport<variableexportvariable=variable=valueexportvariableexportvariable:=variable:=valueexportvariableexportvariable+=variable+=valueexportvariablecdsubdir&&$(MAKE)。還有一個(gè)在“嵌套執(zhí)行”中比較有用的參數(shù),“-w”或是“--print-directory 是“/home/hchen/gnu/make”,如果我們使用“make-w”來(lái)執(zhí)行,那么當(dāng)進(jìn)入該 make:Enteringdirectory而在完成下層make后離 make:Leavingdirectory如果Makefile中出現(xiàn)一些相同命令序列,那么我們可以為這些相同令序列定義一definerun-yaccyacc$(firstword$^)mvy.tab.c$@這里,“run-yacc”是這個(gè)命令包的名字,其不要和Makefile中的變量重名。在“define”和“endef”中的兩行就是命令序列。這個(gè)命令包中的第一個(gè)命令是運(yùn)行Yaccfoo.c:makeMakefileC/C++所不同的標(biāo)”,“命令”或是Makefile的其它部分中。是三個(gè)不同的變量名。傳統(tǒng)的Makefile的變量名是全大寫名方式,但我推薦使用大小變量在時(shí)需要給予初值,而在使用時(shí),需要給在變量名前加上“$”符號(hào),但最好objects=program.ofoo.outils.oprogram:$(objects)cc-oprogram$(objects):foo=cprog.o:$(foo)$(foo)-$(foo)prog.o:prog.ccc-cprog.cfoo=$(bar)bar=$(ugh)ugh=Huh?echo我們執(zhí)行“makeall”將會(huì)打出變量$(foo)的值是“Huh?”($(foo)的值是“Huh?”CFLAGS=$(include_dirs)-Oinclude_dirs=-Ifoo-IbarCFLAGS=$(CFLAGS)-A=B=x:=y:=$(x)barx:=latery:=foobarx:=latery:=$(x)barx:=foo上面都是一些比較簡(jiǎn)單的變量使用了,讓我們來(lái)看一個(gè)復(fù)雜的例子,其中包括了ifeqcur-dir:=$(swhoami:=$(swhoami)host-type:=$(sarch)MAKE:=${MAKE}host-type=${host-type}whoami=${whoami}make”量會(huì)記錄了我們的當(dāng)前Makefile的調(diào)用層數(shù)。nullstringspace:=$(nullstring)#endofthedir:=/foo/bar#directorytoputthefrobs定別的——“$(dir)/file”那么就了。FOO?=ifeq($(originFOO),FOO=barfoo:=a.ob.oc.obar:=$(foo:.o=.c)有以“.o”字串“結(jié)尾”全部替換成“.c”,所以我們的“$(bar)”的值就是“a.cb.c另外一種變量替換的技術(shù)是以“靜態(tài)模式”(參見(jiàn)前面章節(jié))foo:=a.ob.oc.obar:=$(foo:%.o=%.c)樣讓$(bar)變量的值為“a.cb.cc.c”。x=yy=a:=x=y=z=a:=x=y=z=a:=x=variable1variable2:=oy=$(substz=a:=這個(gè)例子中,“$($($(z)))”擴(kuò)展為“$($(y))”,而其再次被擴(kuò)展為“$($(subst1,2,$(x)))”。$(x)的值是“variable1”,substvariable1”中的所有“1”字first_second=oa=firstb=secondall=$($a_$b)a_objects:=a.ob.oc.o1_objects:=1.o2.osources:=這個(gè)例子中,如果$(a1)的值是“a”的話,那么,$(sources)的值就是“a.cb.cc.c”;如果$(a1)的值是“1”,那么$(sources)的值是“1.c2.c3.c”。ifdefdo_sortfunc:=sortfunc:=stripbar:=adbgqfoo:=$($(func)的值就是“abcdgq”,而如果沒(méi)有定義“do_sort”,那么:foo$(sortadbgqc),調(diào)用的就是strip函數(shù)。dir=$(dir)_sources:=$(wildcard$(dir)/*.c)define$(dir)_printobjects=main.ofoo.obar.outils.oobjects+=another.oobjects=main.ofoo.obar.outils.oobjects:=$(objects)another.ovariable:=valuevariable+=morevariable:=variable:=$(variable)variable=valuevariable+=五、overrideoverride<variable>=<value>override<variable>:=override<variable>+=<moreoverridedefinefoo鍵開(kāi)頭,那么make就不會(huì)把其認(rèn)為是命令。definetwo-linesechofooechoMakefile中使用這個(gè)變量了。這對(duì)于我們使用統(tǒng)一的編譯參數(shù)有比較大的好處。如果置的變量會(huì)被傳遞。而定義在文件中的變量,如果要向下層Makefile傳遞,則需要使用exprot關(guān)鍵字來(lái)。(參見(jiàn)前面章節(jié))當(dāng)然,我并不推薦把許多的變量都定義在系統(tǒng)環(huán)境中,這樣,在我們執(zhí)行不用的前面我們所講的在Makefile中定義的變量都是“全局變量”,在整個(gè)文件,我們都可<target...>:<variable-<target...>:overide<variable-prog:CFLAGS=-prog:prog.ofoo.o$(CC)$(CFLAGS)prog.ofoo.oprog.o:$(CC)$(CFLAGS)foo.o:$(CC)$(CFLAGS)bar.o:$(CC)$(CFLAGS)規(guī)則中(prog.ofoo.obar.o的規(guī)則,$(CFLAGS)的值都是“-g”Variable%.o:CFLAGS=-<pattern...>:<variable-<pattern...>:override<variable-libs_for_gcc=-lgnunormal_libs=foo:$(objects)ifeq$(CC)-ofoo$(objects)$(libs_for_gcc)$(CC)-ofoo$(objects)$(normal_libs)表達(dá)式都應(yīng)該以endif結(jié)束。foo:$(CC)-ofoo$(objects)比如“cc”foo:$(CC)-ofoo$(objects)libs_for_gcc=-lgnunormal_libs=ifeq($(CC),gcc)foo:$(CC)-ofoo$(objects)ifeq(<arg1>,<arg2>)ifeq'<arg1>''<arg2>'ifeq"<arg1>""<arg2>"ifeq"<arg1>"'<arg2>'ifeq'<arg1>'ifeq($(strip這個(gè)示例中使用了“strip”函數(shù),如果這個(gè)函數(shù)的返回值是空(Empty),那么ifneq(<arg1>,<arg2>)ifneq'<arg1>''<arg2>'ifneq"<arg1>""<arg2>"ifneq"<arg1>"'<arg2>'ifneq'<arg1>'ifdef<variable-如果變量<variable-name>的值非空,那到表達(dá)式為真。否則,表達(dá)式為假。當(dāng)然,barfoo=$(bar)ifdeffoofrobozz=yesfrobozz=nofoo=frobozz=yesfrobozz=noifndef<variable-有智能。make$(<function>${<function>而不是“$(substa,b,${x})”的形式。因?yàn)榻y(tǒng)一會(huì)更清楚,也會(huì)減少一些不必要的麻煩。comma:=,space:=$(empty)foo:=abbar:=$(subst$(subst名稱:字符串替換函數(shù)——subst$(substee,EE,feetonthe把“feetonthestreet”中的“ee”替換成“EE”,返回結(jié)果是“fEEton$(patsubst<pattern>,<re$(patsubst%.c,%.o,x.c.c把字串“x.c.cbar.c”符合模式[%.c]的單詞替換成[%.o],返回結(jié)果是“x.c.o而“$(var:<suffix>=<recement>)”則相當(dāng)于例:objects=foo.obar.o%.$(strip名稱:去空格函數(shù)——strip$(stripabc把字串“abc“abc$(findstring$(findstringa,ab$(findstringa,b$(filtersources:=foo.cbar.cbaz.sugh.hfoo:$(sources)cc$(filter%.c%.s,$(sources))-o$(filterc%.s,$(sources))返回的值是“foo.cbar.cbaz.s”$(filter-outobjects=main1.ofoo.omain2.obar.omains=main1.omain2.o$(filter-out$(mains),$(objectsfoo.obar.o$(sort名稱:排序函數(shù)——sort示例:$(sortfoobarlose)返回“barfoolose備注:sort函數(shù)會(huì)去掉<list>中相同的單詞。$(word名稱:取單詞函數(shù)——word示例:$(word2,foobarbaz)返回值是“bar$(wordlist名稱:取單詞串函數(shù)——wordlist$(wordlist23foobarbaz)返回值是“barbaz$(words示例:$(wordsfoobarbaz)返回值是“3”備注:如果我們要取<text>中最后的一個(gè)單詞,我們可以這樣:$(word$(words<te$(firstword名稱:首單詞函數(shù)——firstword示例:$(firstwordfoobar)返回值是“foo”路徑。于是,我們可以利用這個(gè)搜索路徑來(lái)指定編譯器對(duì)頭文件的搜索路徑參數(shù)CFLAGS,overrideCFLAGS+=$(patsubst%,-I%,$(subst:,如果我們的“$(VPATH)”值是“src:../headers”,那么“$(dir名稱: 函數(shù)——dir $(dirsrc/foo.chacks)返回值是“src/$(notdir名稱:取文件函數(shù)——notdir $(notdirsrc/foo.chacks)返回值是“foo.chacks$(suffix名稱:取后綴函數(shù)——suffix示例:$(suffixsrc/foo.csrc-1.0/bar.chacks)返回值是“.c.c”。$(basename示例:$(basenamesrc/foo.csrc-1.0/bar.chacks)返回值是“src/foosrc-arhacks$(addsuffix示例:$(addsuffixc,foobar)返回值是“foo.cbar.c”$(addprefix示例:$(addprefixsrc/,foobar)返回值是“src/foosrc/bar”$(join名稱:連接函數(shù)——join示例:$(joinaaabbb111222333)返回值是“aaa111bbb222333四、foreachforeach,Makefile$(foreach所組成的整個(gè)字符串(以空格分隔)將會(huì)是foreach函數(shù)的返回值。names:=abcfiles:=$(foreach根據(jù)“$(n)”foreach$(files)的值是“a.ob.oc.od.o注意,foreach<var>參數(shù)是一個(gè)臨時(shí)的局部變量,foreach五、if$(if<condition>,<then-$(if<condition>,<then-part>,<else-,那么call$(callreverse=$(1)foo=$(call那么,foo的值就是“abreverse=$(2)foo=$(callorigin$(origin“command是,在我們的Makefile中,我們可以這樣寫:ifdefifeq"$(originbletch)""environment"bletch=barf,gag,etc.overrideoverride八、ss函數(shù)也不像其它的函數(shù)。顧名思義,它的參數(shù)應(yīng)該就是操作系統(tǒng)S令。它和反引號(hào)“`”是相同的功能。這就是說(shuō),s函數(shù)把執(zhí)行操作系統(tǒng)命令后的輸出作為awk,sedcontents:=$(scatfoo)files:=$(secho*.c)Makefile運(yùn)行時(shí)信息,并且根據(jù)這些信息來(lái)決定,你是讓make繼續(xù)執(zhí)行,還是停止。$(error<textifdef$(errorerroris$(ERROR_001))ERR=$(errorfoundan.PHONY:errerr:;$(ERR)示例一會(huì)在變量ERROR_001定義了后執(zhí)行時(shí)產(chǎn)生error調(diào)用,而示例二則 error$(warning<textke繼續(xù)執(zhí)行。make一般來(lái)說(shuō),最簡(jiǎn)單的就是直接在命令行下輸入make命令,make命令會(huì)找當(dāng)前 Make的相關(guān)參數(shù)我們會(huì)在后續(xù)章節(jié)中講述。 makef”或是“--file”參數(shù)(“--makefile”參數(shù)也行。例如,我們有個(gè)makefile的名字是“hchen.mk”,那么,我們可以這樣來(lái)讓make來(lái)執(zhí)行這個(gè)文件:make–fmake需在make命令后直接跟目標(biāo)的名字就可以完成(如前面提到的“makeclean”形式)sources=foo.cifneq($(MAKECMDGOALS),clean)include$(sources:.c=.d)基于上面的這個(gè)例子,只要我們輸入令不是“makeclean”,那么makefile會(huì)自動(dòng)包含“foo.d”和“bar.d”這兩個(gè)makefile。.PHONY:all:prog1prog2prog3“prog2“prog3“prog4”,我們可以使用“makeall”make”等功能。我們可以參照這種規(guī)則來(lái)書(shū)寫我們的makefile中的目標(biāo)。但是,GNU搞出這些東西一定有其可取之處(等你的UNIX下的程序文件一多時(shí)你就會(huì)令,或是執(zhí)行的序列。于是我們可以使用make命令的下述參數(shù):“-打印出來(lái),但不執(zhí)行,這些參數(shù)對(duì)于我們調(diào)試makefile很有用處?!?這個(gè)參數(shù)的意思就是把目標(biāo)文件的時(shí)間更新,但不更改目標(biāo)文件。也就是說(shuō),make“-“-W,Make過(guò)其它產(chǎn)商的make的具體參數(shù)還是請(qǐng)參考各自的產(chǎn)品文檔。“-“-“-認(rèn)為所有的目標(biāo)都需要更新(重編譯“-C指定makefile的。如果有多個(gè)“-C”參數(shù),make的解釋是后面的路徑以前面的作為相對(duì)路徑,并以最后的作為被指定。如:“makeC~hchen/testCprog”等價(jià)于“make–C~hchen/test/prog”。“-“-“-“-“-I<dir>” “-j指同時(shí)運(yùn)行命令的個(gè)數(shù)。如果沒(méi)有這個(gè)參數(shù),make“-“-lmake“-“-o“-命令。如果你想查看執(zhí)行makefile前的預(yù)設(shè)變量和規(guī)則,你可以使用“make–p–fmakefile“-“-“-“-“-“-“- “-“-WMakefile“隱含規(guī)則”也就是一種慣例,make本就不用寫出來(lái),make會(huì)自動(dòng)推導(dǎo)出這種規(guī)則,并生成我們需要的[.o]文件。證我們Makefile的兼容性。make先約定好的一些東西。例如,我們有下面的一個(gè)Makefile:foo:foo.occ–ofoofoo.obar.o$(CFLAGS)make果找不到,那么就會(huì)報(bào)錯(cuò)。在上面的那個(gè)例子中,make[.o]foo.o:cc–cfoo.c$(CFLAGS)bar.o:bar.ccc–cbar.cmake越被經(jīng)常使用的,所以,這會(huì)導(dǎo)致我們有些時(shí)候即使我們顯示地指定了目標(biāo)依賴,ae也不會(huì)管。如下面這條規(guī)則(沒(méi)有命令:foo.o:依賴文件“foo.p”(Pascal程序的源文件)有可能變得沒(méi)有意義。如果 foo.o文件。因?yàn)?,在隱含規(guī)則中,Pascal的規(guī)則出現(xiàn)在C的規(guī)則之后,所以,make找到定義在目標(biāo).SUFFIXES的依賴目標(biāo)),那么隱含規(guī)則就會(huì)生效。默認(rèn)的后綴列表是:.out,.a,.ln,.o,.c,.cc,.C,.p,.f,.F,.r,.y,.l,.s,.S,.mod,.sym,“<n>.o”的目標(biāo)的依賴目標(biāo)會(huì)自動(dòng)推導(dǎo)為“<n>.c”,并且其生成命令是“$(CC)$(CPPFLAGS)“<n>.o”的目標(biāo)的依賴目標(biāo)會(huì)自動(dòng)推導(dǎo)為“<n>.p”,并且其生成命令是“$(PC)–“.f”“$(FC)–c“.F”“$(FC)–c$(FFLAGS)“.f”“$(FC)–c$(FFLAGS)Ratfor或有預(yù)處理的Fortran程序到一個(gè)標(biāo)準(zhǔn)的Fortran程序。其使用令是:“.F”“$(FC)–F$(CPPFLAGS)$(FFLAGS)”“.r”“$(FC)–F$(FFLAGS)“<n>.sym”的目標(biāo)的依賴目標(biāo)會(huì)自動(dòng)推導(dǎo)為“<n>.def”,并且其生成命令是:“$(M2C)$(M2FLAGS)$(DEFFLAGS)”n.o>”的目標(biāo)的依賴目標(biāo)會(huì)自動(dòng)推導(dǎo)為“<n>.mod“$(M2C)$(M2FLAGS$(MODFLAGS)“<n>.o”n>.s”,默認(rèn)使用編譯品“as”,并且其生成命令是:“$(AS$(ASFLAGS)”?!?lt;n>.s”的目標(biāo)的依賴目標(biāo)會(huì)自動(dòng)推導(dǎo)為“<n>.SC“cpp”,并且其生成命令是:“$(AS)$(ASFLAGS)”?!?lt;n>”目標(biāo)依賴于“<n>.o”,通過(guò)運(yùn)行C的編譯器來(lái)運(yùn)行程序生成(一般是“l(fā)d”x:y.occ-cx.c-occ-cy.c-occ-cz.c-oz.occx.oy.oz.o-oxrm-fx.orm-fy.orm-f“<n>.c”的依賴文件被自動(dòng)推導(dǎo)為“n.y”(Yacc生成的文件),其生成命令是:“$(YACC)$(YFALGS(“Yacc”是一個(gè)語(yǔ)法分析器,關(guān)于其細(xì)節(jié)請(qǐng)查看相關(guān)資料)生成的文件生成的文件)-在隱含規(guī)則中令中,基本上都是使用了一些預(yù)先設(shè)置的變量。你可以在你的例如,第一條隱含規(guī)則——編譯C程序的隱含規(guī)則令是“$(CC)–c把變量“$(CFLAGS)”重定義成“-g”,那么,隱含規(guī)則中令全部會(huì)以“gcc–c-g匯編語(yǔ)言編譯程序。默認(rèn)命令是“as”RatforCWebC。默認(rèn)命令是“ctangle”。刪除文件命令。默認(rèn)命令是“rmf”再調(diào)用C編譯的隱含規(guī)則最終由[.c]生成[.o]文件,達(dá)到目標(biāo)。標(biāo)會(huì)這樣生成?怎么我的mak
溫馨提示
- 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒(méi)有圖紙預(yù)覽就沒(méi)有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫(kù)網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 2025年食堂承包經(jīng)營(yíng)廢棄物處理與資源化利用合同3篇
- 2025版門衛(wèi)人員招聘與培訓(xùn)服務(wù)合同樣本4篇
- 2025年度消防系統(tǒng)安全評(píng)估與整改合同3篇
- 2024食品安全保密協(xié)議:食品添加劑生產(chǎn)與保密合同3篇
- 模具租賃及后續(xù)加工定制服務(wù)合同2025年版3篇
- 2024年項(xiàng)目投資合同:共擔(dān)風(fēng)險(xiǎn)3篇
- 2025年度租賃權(quán)附帶智能家居安裝合同3篇
- 2024知名品牌家電銷售代理合同
- 2025版公共廣場(chǎng)綠化管理與景觀維護(hù)服務(wù)合同4篇
- 二零二五版貨車租賃與智能物流服務(wù)合同3篇
- 2025-2030年中國(guó)草莓市場(chǎng)競(jìng)爭(zhēng)格局及發(fā)展趨勢(shì)分析報(bào)告
- 奕成玻璃基板先進(jìn)封裝中試線項(xiàng)目環(huán)評(píng)報(bào)告表
- 廣西壯族自治區(qū)房屋建筑和市政基礎(chǔ)設(shè)施全過(guò)程工程咨詢服務(wù)招標(biāo)文件范本(2020年版)修訂版
- 人教版八年級(jí)英語(yǔ)上冊(cè)期末專項(xiàng)復(fù)習(xí)-完形填空和閱讀理解(含答案)
- 2024新版有限空間作業(yè)安全大培訓(xùn)
- GB/T 44304-2024精細(xì)陶瓷室溫?cái)嗔炎枇υ囼?yàn)方法壓痕(IF)法
- 年度董事會(huì)工作計(jì)劃
- 五年級(jí)上冊(cè)口算練習(xí)400題及答案
- 高三數(shù)學(xué)寒假作業(yè)1
- 1例左舌鱗癌手術(shù)患者的圍手術(shù)期護(hù)理體會(huì)
- (完整)100道兩位數(shù)加減兩位數(shù)口算題(難)
評(píng)論
0/150
提交評(píng)論