条件语句可以根据一个变量的值来控制make执行或者忽略Makefile的特定部分。条件语句可以是两个不同变量、或者变量和常量值的比较。要注意的是:条件语句只能用于控制make实际执行的makefile文件部分,它不能控制规则的shell命令执行过程。Makefile中使用条件控制可以做到处理的灵活性和高效性。

7.1      一个例子

首先我们来看一个使用条件判断的Makefile例子;对变量“CC”进行判断,其值如果是“gcc”那么在程序连接时使用库“libgnu.so”或者“libgnu.a”,否则不链接任何库。Makefile中的条件判断部分如下:

……

libs_for_gcc = -lgnu

normal_libs =

……

foo: $(objects)

ifeq ($(CC),gcc)

$(CC) -o foo $(objects) $(libs_for_gcc)

else

$(CC) -o foo $(objects) $(normal_libs)

endif

……

例子中,条件语句中使用到了三个关键字:“ifeq”、“else”和“endif”。其中:

1.        “ifeq”表示条件语句的开始,并指定了一个比较条件(相等)。之后是用圆括号括包围的、使用逗号“,”分割的两个参数,和关键字“ifeq”用空格分开。参数中的变量引用在进行变量值比较时被展开。“ifeq”之后就是当条件满足make需要执行的,条件不满足时忽略。

2.        “else”之后就是当条件不满足时的执行部分。不是所有的条件语句都需要此部分。

3.        “endif”表示一个条件语句的结束,任何一个条件表达式都必须以“endif”结束。

通过上边的例子我们可以了解到。Makefile中,条件表达式工作于文本级别(条件判断处理为文本级别的处理过程),条件的解析是由make来完成的。make是在读取并解析Makefile时根据条件表达式忽略条件表达式中的某一个文本行,解析完成后保留的只有表达式满足条件所需要执行的文本行。上例,make处理条件的过程:

当变量“CC”的值为“gcc”时,整个条件表达式等效于:

foo: $(objects)

$(CC) -o foo $(objects) $(libs_for_gcc)

当变量“CC”值不等于“gcc”时等效于:

foo: $(objects)

$(CC) -o foo $(objects) $(normal_libs)

上面的例子,一种更简洁实现方式:

libs_for_gcc = -lgnu

normal_libs =

ifeq ($(CC),gcc)

libs=$(libs_for_gcc)

else

libs=$(normal_libs)

endif

foo: $(objects)

$(CC) -o foo $(objects) $(libs)

7.2      条件判断的基本语法

一个简单的不包含“else”分支的条件判断语句的语法格式为:

CONDITIONAL-DIRECTIVE

TEXT-IF-TRUE

endif

表达式中“TEXT-IF-TRUE”可以是若干任何文本行,当条件为真时它就将被make作为需要执行的一部分。当条件为假时,不作为需要执行的一部分。

包含“else”的复杂一点的语法格式为:

CONDITIONAL-DIRECTIVE

TEXT-IF-TRUE

else

TEXT-IF-FALSE

endif

表示了如果条件为真,则将“TEXT-IF-TRUE”作为执行Makefile的一部分,否则将“TEXT-IF-FALSE”作为执行的Makefile的一部分。和“TEXT-IF-TRUE”一样,“TEXT-IF-FALSE”可以是若干任何文本行。

条件判断语句中“CONDITIONAL-DIRECTIVE”对于上边的两种格式都是同样的。可以是以下四种用于测试不同条件的关键字。

7.2.1       关键字“ifeq”

此关键字用来判断参数是否相等,格式如下:

`ifeq (ARG1, ARG2)'

`ifeq 'ARG1' 'ARG2''

`ifeq "ARG1" "ARG2"'

`ifeq "ARG1" 'ARG2''

`ifeq 'ARG1' "ARG2"'

替换展开“ARG1”和“ARG1”后,对它们的值进行比较。如果相同则(条件为真)将“TEXT-IF-TRUE”作为make要执行的一部分,否则将“TEXT-IF-FALSE”作为make要执行的一部分(上边的第二种格式)。

通常我们会使用它来判断一个变量的值是否为空(不是任何字符)。参数值可能是通过引用变量或者函数得到的,因而在展开过程中可能造成参数值中包含空字符(空格等)。一般在这种情况时我们使用make的“strip”函数来对它变量的值进行处理,去掉其中的空字符。格式为:

ifeq ($(strip $(foo)),)

TEXT-IF-EMPTY

endif

这样,即就是在“$(foo)”中存在若干前导和结尾空格,“TEXT-IF-EMPTY”也会被作为Makefile需要执行的一部分。

7.2.2       关键字“ifneq”

此关键字是用来判断参数是否不相等,格式为:

`ifneq (ARG1, ARG2)'

`ifneq 'ARG1' 'ARG2''

`ifneq "ARG1" "ARG2"'

`ifneq "ARG1" 'ARG2''

`ifneq 'ARG1' "ARG2"'

关键字“ifneq”实现的条件判断语句和“ifeq”相反。首先替换并展开“ARG1”和“ARG1”,对它们的值进行比较。如果不相同(条件为真)则将“TEXT-IF-TRUE”作为make要执行的一部分,否则将“TEXT-IF-FALSE”作为make要执行的一部分。

7.2.3       关键字“ifdef”

关键字“ifdef”用来判断一个变量是否已经定义。格式为:

`ifdef VARIABLE-NAME'

如果变量“VAEIABLE_NAME”的值非空(在Makefile中没有定义的变量的值为空),那么表达式为真,将“TEXT-IF-TRUE”作为make要执行的一部分。否则,表达式为假,如果存在“TEXT-IF-FALSE”,就将它作为make要执行一部分。当一个变量没有被定义时,它的值为空。“VARIABLE-NAME”可以是变量或者函数的引用。

对于“ifdef”需要说明的是:ifdef只是测试一个变量是否有值,不会对变量进行替换展开来判断变量的值是否为空。对于变量“VARIABLE-NAME”,除了“VARIABLE-NAME=”这种情况以外,使用其它方式对它的定义都会使“ifdef”返回真。就是说,即使我们通过其它方式(比如,定义它的值引用了其它的变量)给它赋了一个空值,“ifdef”也会返回真。我们来看一个例子

例1:

bar =

foo = $(bar)

ifdef foo

frobozz = yes

else

frobozz = no

endif

例2:

foo =

ifdef foo

frobozz = yes

else

frobozz = no

endif

例1中的结果是:“frobozz = yes”;而例2的结果是:“frobozz = no”。其原因就是在例1中,变量“foo”的定义是“foo = $(bar)”。虽然变量“bar”的值为空,但是“ifdef”判断的结果是真。因此当我们需要判断一个变量的值是否为空的情况时,需要使用“ifeq”(或者“ifneq”)而不是“ifdef”。可参考前两个小节的内容。

7.2.4       关键字“ifndef”

关键字“ifndef”实现的功能和“ifdef”相反。格式为:

`ifdef VARIABLE-NAME'

这个就不详细讨论了,它的功能就是实现了和“ifdef”相反的条件判断。

在“CONDITIONAL-DIRECTIVE”这一行上,可以以若干个空格开始,make处理时会被忽略这些空格。但不能以[Tab]字符做为开始(不然就被认为是命令)。条件判断语句中,在除关键字(包括“endif”)之前、条件表达式参数中之外的其他任何地方都可以使用多个空格或[Tab]字符,它不会影响条件判断语句的功能。同样行尾也可以使用注释(“#”开始直到一行的结束)。“else”和“endif”也是条件判断语句的一部分。在书写时它们都是没有任何参数的,可以以多个空格开始(同样不能以[Tab]字符开始)多个空格或[Tab]字符结束。行尾同样可以有注释内容。

在make读取makefile文件时计算表达式的值,并根据表达式的值决定判断语句中那一部分被作为此Makefile所要执行的内容(选择符合条件的语句)。因此在条件表达式中不能使用自动化变量,自动化变量在规则命令执行时才有效。更不能将一个完整的条件判断语句分写在两个不同的makefile文件中,在一个makefile文件使用指示符“include”包含另外一个makefile文件。

7.3      标记测试的条件语句

我们可以使用条件判断语句、变量“MAKEFLAGS”和函数“findstring”,实现对make命令行选项的测试。看一个例子:

archive.a: ...

ifneq (,$(findstring t,$(MAKEFLAGS)))

+touch archive.a

+ranlib -t archive.a

else

ranlib archive.a

endif

这个条件语句判断make的命令行参数中是否包含“-t”(用来更新目标文件的时间戳)。根据命令行参数情况完成对“archive.a”执行不同的操作。命令行前的“+”的意思是告诉make,即使make使用了“-t”参数,“+”之后的命令都需要被执行。

[转] Makefile的条件执行的更多相关文章

  1. 《GNU_makefile》第七章——makefile的条件执行

    条件执行即,通过变量的值,来控制make的执行和忽略. 条件执行只能控制makefile的make语法部分,不能控制shell部分 1.一个例子 - libs_for_gcc = -lgnu norm ...

  2. 成为 Linux 终端高手的七种武器 之七 条件执行&&

    7.条件执行 Bash 也可以连续执行两条命令。 第二条命令仅在第一条命令成功执行后才会开始执行。如要如此,你可以通过键入“&&”,也就是两个“&”字符进行分隔,在同一行输入两 ...

  3. 多文件目录下makefile文件递归执行编译所有c文件

    首先说说本次嵌套执行makefile文件的目的:只需make根目录下的makefile文件,即可编译所有c文件,包括子目录下的. 意义:自动化编译行为,以后编译自己的c文件时可把这些makefile文 ...

  4. ARM指令集——条件执行、内存操作指令、跳转指令

    ARM 汇编指令条件执行 在ARM模式下,任何一条数据处理指令可以选择是否根据操作的结果来更新CPSR寄存器中的ALU状态标志位.在数据处理指令中使用S后缀来实现该功能. 不要在CMP,CMN,TST ...

  5. 【转】多文件目录下makefile文件递归执行编译所有c文件

    首先说说本次嵌套执行makefile文件的目的:只需make根目录下的makefile文件,即可编译所有c文件,包括子目录下的. 意义:自动化编译行为,以后编译自己的c文件时可把这些makefile文 ...

  6. 【学习总结】GirlsInAI ML-diary day-10-if条件执行

    [学习总结]GirlsInAI ML-diary 总 原博github链接-day10 认识if条件执行 一般条件执行 分支执行 链式条件执行 嵌套条件执行 1-if一般条件执行 格式:如果(满足这个 ...

  7. if、else、else if后的条件执行体

    if.else.else if后的条件执行体有两种情况:一种是用大括号“{ }”括起来的代码块,这个代码块被称作条件执行体:另一种是以分号“:”作为结束符的单行语句或空语句(仅有一个分号),这个单行语 ...

  8. 什么时候出现死锁,如何解决?mysql 引擎? 多个like or 查询sql如何优化?什么是常量池?for条件执行顺序

    1. 什么时候出现死锁,如何解决?mysql 引擎? 多个like or 查询sql如何优化? 资源竞争导致死锁,比如A B 同时操作两条记录,并等待对方释放锁. 优化sql, 加缓存,主从(如读写分 ...

  9. Java编程的逻辑 (9) - 条件执行的本质

    本系列文章经补充和完善,已修订整理成书<Java编程的逻辑>,由机械工业出版社华章分社出版,于2018年1月上市热销,读者好评如潮!各大网店和书店有售,欢迎购买,京东自营链接:http:/ ...

随机推荐

  1. arp断网攻击解决办法

    局域网中有这个提示arp断网攻击是正常的,说明防火墙已经拦截了,是有人用P2P工具控制你的网速,或者是局域网有机器中病毒了也会有这样的提示,不过不用担心,今天给大家带来几个防止arp断网攻击的办法,希 ...

  2. Java简介(1)

    起源 略. 组成 Java由四方面组成: 1.Java编程语言 2.Java文件格式 3.Java虚拟机(JVM) 4.Java应用程序接口(Java api) 体系 JavaSE , JavaEE, ...

  3. ecshop里Ajax.call()方法定义

    Ajax.call()在哪个文件中定义的? 在加载的js/transport.js文件里面. Ajax.cal()方法就是Transport.run()方法

  4. Day3 set集合、函数和装饰器

    set特性:无序,不重复,可嵌套 创建setset_example = set("123", "213", "234", "432 ...

  5. type和instance

    获取对象类型 type(object) >>> test_data = [1, 2, 3] >>> type(test_data) <type 'list'& ...

  6. SCJP_104——题目分析(4)

    14. which three are valid declaraction of a float? ADFA. float foo=-1; B. float foo=1.0; C. float fo ...

  7. 关于 从别人电脑上 高版本的 Xcode上拷贝过来的项目的 不能运行模拟器的 解决方法

    如图 从别人电脑上 拷贝过来的  工程  打开后  点击 iOS  Device  只有  一个选项  没有模拟器.这说明 自己的 Xcode 的版本比 创建这个工程所用的版本低.所以 要睇啊你tar ...

  8. (未解决)问题记录ionic android 签名之后安装到手机上点击运行出现闪退,不签名运行正常

    Log日志如下: - ::): error opening trace ) - ::): FATAL EXCEPTION: main - ::): java.lang.RuntimeException ...

  9. h.264 FMO

    在H.264之前的标准中,比如H.263,其比特流中的数据是按照一个宏块接一个宏块的方式排列的,一旦发生丢包,很多相邻宏块信息都会丢失,很难进行错误隐藏处理.在H.264中加入了一项新特性:把宏块在比 ...

  10. 值得IT运维人员警示的“一件事儿”

    昨天,一个用户打来了紧急求助电话,并且发了邮件,弄得我当时紧张了一下,以为他们那里又出了什么乱子.用户在电话里说:应用系统性能很差,运行很慢,几近“卡死”的感觉,而且重启了多次应用和数据库服务器,最终 ...