如何编写Makefile(3)-书写命令

目录

一、显示命令

二、命令执行

三、命令出错

四、嵌套执行

五、定义命令包


        我们继续Makefile编写的内容学习~第二节可参考:↓传送门↓

如何编写Makefile(2)-CSDN博客icon-default.png?t=O83Ahttps://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就可以,后面不需要书写其它内容,表示传递所有变量。但是有两个变量,SHELLMAKEFLAGS,无论是否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在执行命令包时,命令包中的每个命令会被依次独立执行。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

猿核试Bug愁

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值