千锋嵌入式学院最专业的嵌入式培训机构简介一、构建多个目标二、内部规则三、后缀规则四、用make管理程序库五、Makefile和子目录六、GNUmake和gcc的有关选项七、小结千锋嵌入式学院最专业的嵌入式培训机构make是Linux下的一款程序自动维护工具,配合makefile的使用,就能够根据程序中模块的修改情况,自动判断应该对那些模块重新编译,从而保证软件是由最新的模块构成。本文分为上下两部分,我们在上一篇文章中分别介绍了make和makefile的一些基本用法,在本文中,我们会对make和makefile的功能做进一步的介绍。千锋嵌入式学院最专业的嵌入式培训机构有时候,我们想要在一个makefile中生成多个单独的目标文件,或者将多个命令放在一起,比如,在下面的示例mymakefile3中我们将添加一个clean选项来清除不需要的目标文件,然后用install选项将生成的应用程序移动到另一个目录中去。这个makefile跟前面的mymakefile较为相似,不同之处笔者用黑体加以标识:all:main#使用的编译器CC=gcc#安装位置INSTDIR=/usr/local/bin#include文件所在位置INCLUDE=.#开发过程中所用的选项CFLAGS=-g-Wall–ansi千锋嵌入式学院最专业的嵌入式培训机构#发行时用的选项#CFLAGS=-O-Wall–ansimain:main.of1.of2.o$(CC)-omainmain.of1.of2.omain.o:main.cdef1.h$(CC)-I$(INCLUDE)$(CFLAGS)-cmain.cf1.o:f1.cdef1.hdef2.h$(CC)-I$(INCLUDE)$(CFLAGS)-cf1.cf2.o:f2.cdef2.hdef3.h$(CC)-I$(INCLUDE)$(CFLAGS)-cf2.cclean:千锋嵌入式学院最专业的嵌入式培训机构-rmmain.of1.of2.oinstall:main@if[-d$(INSTDIR)];\then\cpmain$(INSTDIR);\chmoda+x$(INSTDIR)/main;\chmodog-w$(INSTDIR)/main;\echo“Installedin$(INSTDIR)“;\else\echo“Sorry,$(INSTDIR)doesnotexist”;\fi千锋嵌入式学院最专业的嵌入式培训机构在这个makefile中需要注意的是,虽然这里有一个特殊的目标all,但是最终还是将main作为目标。因此,如果执行make命令时没有在命令行中给出一个特定目标的话,仍然会编译连接main程序。其次要注意后面的两个目标:clean和install。目标clean没有依赖模块,因为没有时间标记可供比较,所以它总被执行;它的实际意图是引出后面的rm命令来删除某些目标文件。我们看到rm命令以-开头,这时即使表示make将忽略命令结果,所以即使没有目标供rm命令删除而返回错误时,makeclean依然继续向下执行。接下来的目标install依赖于main,所以make知道必须在执行安装命令前先建立main。用于安装的指令由一些shell命令组成。千锋嵌入式学院最专业的嵌入式培训机构因为make调用shell来执行规则,并且为每条规则生成一个新的shell,所以要用一个shell来执行这些命令的话,必须添加反斜杠,以使所有命令位于同一个逻辑行上。这条命令用@开头,表示在执行规则前不会向标准输出打印命令。为了安装应用程序,目标install会一条接一条地执行若干命令,并且执行下一个之前,不会检查上一条命令是否成功。若想只有当前面的命令取得成功时,随后的命令才得以执行的话,可以在命令中加入&&,如下所示:@if[-d$(INSTDIR)];\then\cpmain$(INSTDIR)&&\chmoda+x$(INSTDIR)/main&&\chmodog-w$(INSTDIR/main&&\echo“Installedin$(INSTDIR)“;\else\echo“Sorry,$(INSTDIR)doesnotexist”;false;\fi千锋嵌入式学院最专业的嵌入式培训机构这是shell的“与”指令,只有当在前的命令成功时随后的命令才被执行。这里不必关心前面命令是否取得成功,只需注意这种用法就可以了。要想在/usr/local/bin目录安装新命令必须具有特权,所以调用makeinstall命令之前,可以让Makefile使用一个不同的安装目录,或者修改该目录的权限,或切换到root用户。如下所示$rm*.omain$make-fMymakefile3gcc-I.-g-Wall-ansi-cmain.cgcc-I.-g-Wall-ansi-cf1.cgcc-I.-g-Wall-ansi-cf2.cgcc-omainmain.of1.of2.o千锋嵌入式学院最专业的嵌入式培训机构$make-fMymakefile3make:Nothingtobedonefor‘all’.$rmmain$make-fMymakefile3installgcc-omainmain.of1.of2.oInstalledin/usr/local/bin$make-fMymakefile3cleanrmmain.of1.of2.o$千锋嵌入式学院最专业的嵌入式培训机构让我们对此作一简单介绍,首先删除main和所有目标文件程序,由于将all作为目标,所以make命令会重新编译main。当我们再次执行make命令时,由于main是最新的,所以make什么也不做。之后,我们删除main程序文件,并执行makeinstall,这会重新建立二进制文件main并将其复制到安装目录。最后,运行makeclean命令,来删去所有目标程序。千锋嵌入式学院最专业的嵌入式培训机构迄今为止,我们已经能够在makefile中给出相应的规则来指出具体的处理过程。实际上,除了我们显式给出的规则外,make还具有许多内部规则,这些规则是由预先规定的目标、依赖文件及其命令组成的相关行。在内部规则的帮助下,可以使makefile变得更加简洁,尤其是在具有许多源文件的时候。现在以实例加以说明,首先建立一个名为foo.c的C程序源文件,文件内容如下所示:#include#includeintmain(){printf(“HelloWorld\n”);exit(EXIT_SUCCESS);}千锋嵌入式学院最专业的嵌入式培训机构现在让我们用make命令来编译它:$makefooccfoo.c-ofoo$您会惊奇地发现,尽管我们没有指定makefile,但是make仍然能知道如何调用编译器,并且调用的是cc而不是gcc编译器。这在Linux上没有问题,因为cc常常会链接到gcc程序。这完全得益于make内建的内部规则,另外这些内部规则通常使用宏,所以只要为这些宏指定新的值,就可以改变内部规则的默认动作,如下所示:$rmfoo$makeCC=gccCFLAGS=”-Wall-g”foogcc-Wall-gfoo.c-ofoo$千锋嵌入式学院最专业的嵌入式培训机构用make命令加-p选项后,可以打印出系统缺省定义的内部规则。它们包括系统预定义的宏、以及产生某些种类后缀的文件的内部相关行。内部规则涉及的文件种类很多,它不仅包括C源程序文件及其目标文件,还包括SCCS文件、yacc文件和lex文件,甚至还包括Shell文件。当然,我们更关心的是如何利用内部规则来简化makefile,比如让内部规则来负责生成目标,而只指定依赖关系,这样makefile就简洁多了,如下所示:main.o:main.cdef1.hf1.o:f1.cdef1.hdef2.hf2.o:f2.cdef2.hdef3.h千锋嵌入式学院最专业的嵌入式培训机构前面我们已经看到,有些内部规则会根据文件的后缀(相当于Windows系统中的文件扩展名)来采取相应的处理。换句话说,这样当make见到带有一种后缀的文件时,就知道使用哪些规则来建立一个带有另外一种后缀的文件,最常见的是用以.c结尾的文件来建立以.o结尾的文件,即把源文件编译成目标程序,但是不连接。现在举例说明后缀规则的应用。有时候,我们需要在不同的平台下编译源文件,例如Windows和Linux。假设我们的源代码是C++编写的,那么Windows下其后缀则为.cpp。不过Linux使用的make版本没有编译.cpp文件的内部规则,倒是有一个用于.cc的规则,因为在UNIX操作系统中c++文件扩展名通常为.cc。这时候,要么为每个源文件单独指定一条规则,要么为make建立一条新规则,告诉它如何用.cpp为扩展名的源文件来生成目标文件。如果项目中的源文件较多的话,后缀规则就可以派上用场了。要添加一条新后缀规则,首先在makefile文件中加入一行来告诉make新后缀是什么;然后就可以添加使用这个新后缀的规则了。这时,make要用到一条专用的语法:千锋嵌入式学院最专业的嵌入式培训机构.旧后缀名.新后缀名:它的作用是定义一条通用规则,用来将带有旧后缀名的文件变成带有新后缀名的文件,文件名保持不变,如要将.cpp文件编译成.o文件,可以使用一个新的通用规则:.SUFFIXES:.cpp.cpp.o:$(CC)-xc++$(CFLAGS)-I$(INCLUDE)-c$上面的“.cpp.o:”告诉make这些规则用于把后缀为.cpp的文件转换成后缀为.o的文件。其中的标志“-xc++”的作用是告诉gcc这次要编译的源文件是c++源文件。这里,我们使用一个宏$来通指需要编译的文件的名称,不管这些文件名具体是什么。我们只需知道,所有以.cpp为后缀的文件将被编译成以.o为后缀的文件,例如以是app.cpp的文件将变成app.o。千锋嵌入式学院最专业的嵌入式培训机构注意,我们只跟make说明如何把.cpp文件变成.o文件就行了,至于如何从目标程序文件变成二进制可执行文件,因为make早已知晓,所以就不用我们费心了。所以,当我们调用make程序时,它会使用新规则把类似app.cpp这样的程序变成app.o,然后使用内部规则将app.o文件连接成一个可执行文件app。现在,make已经知道如何处理扩展名为.cpp的c++源文件,除此之外,我们还可以通过后缀规则将文件从一种类型转换为另一种类型。不过,较新版本的make包