20230421Makefile语法

一、通配符

“*”:匹配任意数量的字符,包含0个和多个字符
“*.c”:匹配所有以".c"结尾的文件(包括空字符串)
“%.c”:匹配所有以".c"结尾的文件(不包括空字符串)

“?”:匹配单个任意字符
“doc?.txt”:可以匹配"doc1.txt"、"docA.txt"等文件,但无法匹配"doc10.txt"等非单个字符文件

“[characters]”:匹配方括号中任意一个字符
“[ab]*.c”:匹配所有以"a"或"b"开头,且以.c结尾的文件名
“*.[cho]”:匹配所有以"c"、“h”、"o"后缀的文件名

“[^characters]”:匹配除了方括号中指定的字符以外的任意一个字符

注意,makefile中使用通配符时,通配符需要用引号括起来避免被shell扩展成为实际匹配的文件列表,导致执行出错
正常情况下,假设当前文件夹下有三个文件"foo.c"、“bar.c”、“baz.h”
现在执行命令

gcc *.c -o myapp

在执行这个命令前,shell会将通配符"*.c"扩展成文件列表"foo.c"和"bar.c",再将扩展后的命令行参数传给gcc编译器,即等价于

gcc foo.c bar.c -o myapp

这种行为就是通配符被shell扩展成实际匹配的文件列表
特殊情况下

foo:*.c
	gcc *.c -o foo

在这个规则里,当文件目录下只存在一个".c"文件时规则可以正常执行,如果存在多个".c"文件就会执行失败,gcc命令会以"foo.c bar.c"的顺序传入参数,最终编译成目标二进制文件"foo",但因为gcc编译器必须接受一堆具体的文件名作为参数而不是通配符,

所以会导致编译错误,因此需要用引号将通配符括起来避免这种扩展行为,如下方示例

foo:*.c
	gcc "$(wildcard *.c)" -o foo

这样shell就无法对其进行扩展,"$(wildcard *.c)"会返回满足模式的具体文件名列表

SRCS=$(wildcard *.c)
foo:$(SRCS)
	gcc $^ -o $@

在这上述规则中,"$(wildcard *.c)“会获取所有的”.c"文件列表名,并存入变量"SRCS"中,按字母顺序排序。在"foo"规则中,使用

“$(SRCS)“代替”*.c"的方式,可以避免重复定义目标文件的问题。在实际编译过程中,gcc命令会先将”.c"文件进行编译,然后再进行链接生成目标文件

“*.c"与”$(wildcard *.c)"
“*.c"匹配指定目录下(或是当前目录下)所有以”.c"结尾的文件,文件名是固定的,并且顺序是不确定的,直接使用会导致重复定义目标文件从而造成编译失败
“$(wildcard *.c)“表示获取当前文件夹下所有以”.c"后缀的文件,保存在一个变量中,这个变量包含了所有”.c"文件名,并按字母顺序排序。使用会将它们作为完整的目标文件名进行编译,每个".c"文件被视为一个独立的目标文件进行编译,不会导致重复定义目标文件的问题

二、“VPATH"与"vpath”

大型工程中各模块功能代码会存放在不同文件夹中,当make在当前目录找不到文件时候,会根据变量"VPATH"所指定的目录去寻找文件

1、环境变量"YPATH"

例如,将一个C文件与一个位于"/usr/include"目录下的头文件一起编译↓

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值