Makefile 显式规则和隐式规则

最近在使用Makefile的过程中发现了一个很有意思的现象,现在抽象一个简单的例子来展示这个现象

Makefile内容如下

makefile 复制代码
% : %.in
	echo $<
	echo $@

文件列表如下:

makefile 复制代码
total 4.0K
-rw-rw-r-- 1 raid raid  0 Oct 28 18:23 aa.stage.in
-rw-rw-r-- 1 raid raid  0 Oct 28 18:09 cc.sh.in
-rw-rw-r-- 1 raid raid  0 Oct 28 18:23 d.c.in
-rw-rw-r-- 1 raid raid 29 Oct 28 18:09 Makefile

执行:

shell 复制代码
➜  22 make aa.stage
echo aa.stage.in
aa.stage.in
echo aa.stage
aa.stage
➜  22 make cc.sh
make: *** No rule to make target 'cc.sh'.  Stop.
➜  22

可以看到一个执行成功,一个执行失败,出现问题的原因是Makefile的匹配规则

make 的模式匹配算法在处理带有后缀的目标时,会优先寻找更具体的规则

shell 复制代码
➜ make -p 
...
# Not a target:
.w:
#  Builtin rule
#  Implicit rule search has not been done.
#  Modification time never checked.
#  File has not been updated.

# Not a target:
.SUFFIXES: .out .a .ln .o .c .cc .C .cpp .p .f .F .m .r .y .l .ym .yl .s .S .mod .sym .def .h .info .dvi .tex .texinfo .texi .txinfo .w .ch .web .sh .elc .el

...

可以看到Makefile内置了这些后缀,也就是说Makefile认识这些后缀,当 make 看到一个目标文件以这些它认识的后缀结尾时,优先使用显式规则,所以可以看到make cc.sh的时候提示找不到cc.sh,因为确实没有cc.sh这个target,而make aa.stage的时候没问题,因为.stage 后缀Makefile不认识,走的还是模式匹配

通过-R可以取消Makefile使用的这些内置变量规则

go 复制代码
➜  22 make -R cc.sh
echo cc.sh.in
cc.sh.in
echo cc.sh
cc.sh

可以看到能成功执行了

相关推荐
追逐时光者3 小时前
一款使用 C# 编写专为 Windows 11 打造的文件资源管理器增强工具!
后端·.net
风象南4 小时前
普通人用AI加持赚到的第一个100块
人工智能·后端
冰_河5 小时前
QPS从300到3100:我靠一行代码让接口性能暴涨10倍,系统性能原地起飞!!
java·后端·性能优化
JavaGuide8 小时前
7 道 RAG 基础概念知识点/面试题总结
前端·后端
桦说编程8 小时前
从 ForkJoinPool 的 Compensate 看并发框架的线程补偿思想
java·后端·源码阅读
格砸9 小时前
从入门到辞职|从ChatGPT到OpenClaw,跟上智能时代的进化
前端·人工智能·后端
蝎子莱莱爱打怪10 小时前
GitLab CI/CD + Docker Registry + K8s 部署完整实战指南
后端·docker·kubernetes
哈密瓜的眉毛美10 小时前
零基础学Java|第三篇:DOS 命令、转义字符、注释与代码规范
后端
用户605723748730810 小时前
AI 编码助手的规范驱动开发 - OpenSpec 初探
前端·后端·程序员