makefile--隐式规则


1、makefile中出现同名目标时

  • 同名依赖:所有的依赖将合并在一起,成为目标的最终依赖
  • 同名命令:当多处出现同一目标的命令时,make发出警告;所有之前定义的命令被最后定义的命令取代
    示例1--makefile中出现同名依赖
.PHONY : all

all : a.txt

all : b.txt

a.txt :
    @echo "this is $@"

b.txt :
    @echo "this is $@"
14--1.PNG

示例2--makefile中出现同名命令

.PHONY : all

all :
    @echo "command-1"

all : 
    @echo "command-2"

all :
    @echo "this is $@"
14-2.PNG

注意事项

  • 当使用include关键字包含其它文件时,需要确保被包含文件中的同名目标只有依赖,没有命令;否则,同名目标的命令将会被最后定义的命令取代。

示例3--使用include关键字时,出现同名命令
makefile文件与1.mk文件位于同一个目录下

makefile

.PHONY : all

all :
    @echo "this is $@"

include 1.mk

1.mk

all :
    @echo "this command from 1.mk"
14-3.PNG

2、隐式规则(build-in rules)

  • make提供了一些常用的,例行的规则实现
  • 当相应目标的规则未提供时,make尝试使用隐式规则
    示例4--目标的规则未提供,make使用隐式规则
    func.h
#ifndef FUNC_H
#define FUNC_H

#define HELLO "hello world"

void foo();

#endif

func.c

#include "stdio.h"
#include "func.h"

void foo()
{
    printf("void foo():%s\n",HELLO);    
}

main.c

#include "stdio.h"
#include "func.h"

extern void foo();

int main()
{
    foo();

    return 0;
}

makefile

SRCS := $(wildcard *.c)
OBJS := $(SRCS:.c=.o)

app.out : $(OBJS)
    $(CC) -o $@ $^
    $(RM)
    @echo "Target ===> $@"

makefile等价于下面的文件

SRCS := $(wildcard *.c)
OBJS := $(SRCS:.c=.o)

app.out : $(OBJS)
    $(CC) -o $@ $^
    $(RM)
    @echo "Target ===> $@"  

%.o : %.c
    @echo "rule"
    $(CC) -c -o $@ $^
14-4.PNG

从以上makefile执行结果可以看到,CC、RM、以及%.o : %.c 调用的(CC) -c -o@ $^,都是make中默认的隐式规则。

通过使用$(.VARIABLES)可以查看make中所有预定义的变量

隐式规则小结一

  • 当make发现目标的依赖不存在时,
    • 尝试通过依赖名逐一查找隐式规则;
    • 并且通过依赖名推导出可能需要的源文件。


      隐式规则图示.png

示例5--获取makefile中的预定义变量

all :
    @echo "$(.VARIABLES)"
14-5.PNG

示例6--改变预定义变量将改变隐式规则的行为

SRCS := $(wildcard *.c)
OBJS := $(SRCS:.c=.o)
CC := gcc
RM := rm -rf *.o

app.out : $(OBJS)
    $(CC) -o $@ $^
    $(RM)
    @echo "Target ===> $@"  

%.o : %.c
    $(CC) -o $@ -c $^
14-6.PNG

隐式规则小结二

  • make提供了生成目标文件的隐式规则
  • 隐式规则会使用预定义变量完成编译工作
  • 改变预定义变量将部分改变隐式规则的行为
  • 当存在自定义规则时,不再使用隐式规则

隐式规则副作用

  • 编译行为难以控制
    • 大量使用隐式规则可能产生意想不到的编译行为
  • 编译效率低下
    • make从隐式规则和自定义规则中选择最终使用的规则
  • 隐式规则链
    • 当依赖的目标不存在时,make会极力组合各种隐式规则对目标进行创建,进而产生意料之外的编译行为。例如:需要名为N.o的目标,隐式规则创建过程为:N.y=>N.c=>N.o

查看make的隐式规则

  • 查看所有隐式规则:make -p
  • 查看具体隐式规则:make -p | grep "XXX"
    示例7,命令行输入查看%.o默认的所有隐式规则make -p | grep "%.o",隐式规则的查找从上往下依次进行匹配
14-8.PNG

3、隐式规则的禁用

  • 局部禁用(用于makefile文件内部)
    • 在makefile中自定义规则
%.o : %.c
    $(CC) -o $@ -c $^
-   在makefile中定义模式为空(如:%.o:%c)
%.o : %.p
    
  • 全局禁用(用于命令行编译)
    • make -r

小结

  • 隐式规则可能造成意想不到的编译行为
  • 在实际工程项目中尽量不使用隐式规则
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容