目录
我们继续Makefile编写的内容学习~第二节可参考:↓传送门↓
如何编写Makefile(2)-CSDN博客
https://blog.csdn.net/L_peanut/article/details/144189831?spm=1001.2014.3001.5502 本节内容主要介绍Makefile的书写命令。
Makefile中每条规则的命令和操作系统Shell的命令行都是一致的,make在执行过程中是按照顺序逐条执行的,每条命令必须以Tab键开头,除非命令是在依赖规则后面的分号后书写的。命令行中的空格或空行会被忽略,但是如果空格或空行是以Tab键开头,则会被make认为是空命令。
一、显示命令
一般情况下,make会把其要执行的命令行在命令执行前输出到屏幕上。当我们在命令行前使用“@”字符时,这个命令将不被make显示出来,如:
@echo 正在编译xxx……
当make执行时,会输出“正在编译xxx……”字符串,但不会输出命令,如果没有“@”,则make会输出:
echo 正在编译xxx……
正在编译xxx……
如果make执行时,添加了make参数-n或--just-print,那么只显示命令,但不会执行命令,这个功能对我们调试Makefile非常有用,可以查看我们书写的命令执行时是什么样子或者执行顺序。
make的参数-s或--silent或--quiet则是全面禁止命令的显示。
二、命令执行
当依赖目标的文件时间新于目标时,也就是当规则的目标需要被更新时,make会逐条执行气候的命令。如果需要让上一条命令的结果应用在下一条命令中时,应该使用分号分隔这两条命令。比如第一条命令是cd命令,你希望第二条命令在cd的基础上运行,就不可以将这两条命令写在两行上,而应该写在同一行,用分号隔开。例如:
#例1
exec:
cd /home/user
pwd
#例2
exec:
cd /home/user;pwd
当执行make exec时,例1中的cd没有作用,pwd会打印当前的Makefile目录;例2中cd起到作用,pwd会打印“/home/user”。
make一般是使用环境变量SHELL中定义的系统Shell来执行命令,默认情况下使用UNIX的标准shell——/bin/sh来执行命令。首先,make会在SHELL所指定的路径中查找命令解释器,如果未找到,将会在当前盘符中的当前目录中寻找,如果还未找到,将会在PATH环境变量所定义的所有路径中寻找。
三、命令出错
当命令执行完后,make会检测每个命令的返回代码,如果命令返回成功,则make会执行下一条命令,当规则中所有命令成功返回后,这个规则便认为是完成了。如果一个规则中的某个命令出错(命令退出码非零),make将会终止执行当前规则,有可能终止所有规则的执行。
有些时候命令出错并不代表是错误的,例如mkdir命令,我们需要创建一个新的目录,如果目录不存在,那么mkdir成功执行,如果目录已经存在,那么就出错。我们使用mkdir的目的是一定要有这样的一个目录,所以不希望mkdir出错而导致规则终止。为了避免这一点,忽略命令的出错,我们可以在Makefile的命令行前面加一个减号“-”(Tab键之后),其作用是无论命令出错与否,都认为是成功的。例如:
clean:
-rm -f *.o
还有一个全局的办法:执行make命令时添加-i或--ignore-errors参数,这样Makefile中的所有命令将会忽略错误。如果一个规则是以.INGORE作为目标的,那么这个规则中的所有命令将会忽略错误。make还有一个参数,-k或--keep-going,作用是如果某个规则中的命令出错了,就终止该规则的执行,但继续执行其它规则。
四、嵌套执行
对于大型工程项目,通常会将不同模块或不同功能的源文件放在不同的目录中,我们可以在每个目录都写一个该目录的Makefile,这有助于让我们的Makefile更加简洁,不至于将所有的内容都写在同一个Makefile中,因为这样会很难维护Makefile。这种方式有助于我们模块编译和分段编译。
例如,有一个子目录subdir,这个目录下有个Makefile文件,指明该目录下文件的编译规则。那么我们总控Makefile可以这样写:
subsystem:
cd subdir && $(MAKE)
等价于:
subsystem:
$(MAKE) -C subdir
定义$(MAKE)宏变量的意义是:或许我们的make需要一些参数,定义成这样有利于维护。这两个例子的意思是先进入“subdir”目录,然后执行make命令。
我们将这个Makefile文件叫做“总控Makefile”,其中的变量可以传递到下级Makefile中(如果显式声明),但是不会覆盖下层的Makefile中所定义的变量,除非指定了-e参数。如果要传递变量到下级Makefile中,可以这样声明:
export <variable ...>;
如果不想让某些变量传递到下级Makefile,可以这样声明:
unexport <variable ...>;
例如:
#例1
export variable=value
#等价于
variable = value
export variable
#等价于
export variable:=value
#等价于
variable:=value
export variable
#例2
export variable+=value
#等价于
variable+=value
export variable
如果要传递所有的变量,只需要一个export就可以,后面不需要书写其它内容,表示传递所有变量。但是有两个变量,SHELL和MAKEFLAGS,无论是否export,这两个变量都会传递到下层Makefile中,尤其是MAKEFLAGS,因为其包含了make的参数信息,如果我们执行“总控Makefile”时有make参数或者是在上层Makefile中定义了这个变量,那么MAKEFLAGS变量将会是这些参数,并传递给下层,这是一个系统级的环境变量。
但是make命令中的几个参数并不往下传递,“-C -f -h -o -W”,如果不想往下层传递参数,可以这样:
subsystem:
cd subdir && $(MAKE) MAKEFLAGS=
还有一个在“嵌套执行”中比较有用的参数,-w或--print-directory,作用是在make的过程中输出一些信息,显示目前的工作目录。例如,如果我们的下级make目录是“/home/user/gnu/make”,如果我们使用make -w来执行,那么进入该目录时,会显示:
make: Entering directory '/home/user/gnu/make'.
在完成下层make后离开目录时,会显示:
make: Leaving directory '/home/user/gnu/make'.
当使用-C参数来指定make下层Makefile时,-w会被自动打开。如果参数中有-s(--slient)或--no-print-directory时,-w总是失效的。
五、定义命令包
如果Makefile中出现一些相同命令序列时,我们可以为这些命令定义一个变量。定义这种命令序列的语法以define开始,以endef接收,例如:
define run-yacc
yacc $(firstword $^)
mv y.tab.c $@
endef
其中“run-yacc”是这个命令包的名字,不要与Makefile中的变量重名。在define和endef中的两行就是命令序列。这个命令包中的第一个命令是运行Yacc程序 ,因为Yacc总会生成“y.tab.c”文件,所以第二行的命令就是把这个文件的名字进行修改。
a.c:a.y
$(run-yacc)
使用这个命令包就像使用变量一样,在这个命令包的使用中,“run-yacc”中的$^就是a.y,$@就是a.c,make在执行命令包时,命令包中的每个命令会被依次独立执行。
-书写命令&spm=1001.2101.3001.5002&articleId=144242622&d=1&t=3&u=2b42baa7f8cb42c2b27d18db9c97da1c)

被折叠的 条评论
为什么被折叠?



