当前位置: 首页 > news >正文

1-7makefile

文章目录

    • 简介
    • 1 规则
      • 1.1规则的执行
    • 2 变量
    • 3 模式匹配
    • tips:规则同名处理机制
    • 4 函数
      • 4.1 wildcard
      • 4.2 patsubst
    • 5 练习
      • 5.1 clean

简介

make 是解释 makefile中指令的工具

1 规则

target,target...:depend1, depend2,...
	command
	command
	...
# 单目标
app:a.c, b.c, c.c
	gcc a.c b.c c.c -o app
	
# 多目标
app1, app2: a.cpp b.cpp
	gcc a.cpp -o app1
	gcc b.cpp -o app2
	
# 规则的嵌套
app: a.o, b.o
	gcc a.o b.o -o app
a.o: a.c
	gcc -c a.c 
b.o: b.c
	gcc -c b.c
  • command: 一般是shell命令,前面需要tab缩进,并且独占一行

  • depend:命令需要的依赖,可以多个,也可以为空

  • target:命令生成的目标,执行命令后可以不生成任何文件,这样的目标是 伪目标

1.1规则的执行

# in makefile
app: depent1
        echo "first -1"
        echo "first -2"

depend:
        "empty"
depent1: 
        echo "depend 1 skip last-empty-depend"
        touch test.md

result:

liyb@lyb:/tmp/test_makefile$ make
echo "depend 1 skip last-empty-depend"
depend 1 skip last-empty-depend
echo "first -1"
first -1
echo "first -2"
first -2
# 并且目录中生成了test.md文件
  • 可以看到,make只执行第一条规则,以及与之嵌套的规则

  • 在执行 command的时候,会先输出command本身,再执行命令

  • 不在第一个规则之中提及的的规则默认是不被执行的

  • 执行指定的一条规则

# make target
# 这里“empty”不是一个shell命令,所以报错,说明确实执行到了
liyb@lyb:/tmp/test_makefile$ make depend
"empty" 
/bin/sh: 1: empty: not found
make: *** [makefile:6:depend] 错误 127

1.2 文件的时间戳

  • 并不是每次make 都会更新目标
  • 如果target的时间戳小于depend (depend更新过),这时候才会重新生成target
  • 另外,如果target 是一个伪目标(depend如果是伪的也是),每次make都会更新,因为没有这个文件/依赖呀

1.3 make工具的自动推导

如果我们写的规则并不严谨,缺乏了某些依赖,在一定程度上它可以帮我们自动生成

例如:依赖中有a.o,但是目录中只有a.c,那么make会使用cc命令自动生成a.o 供目标使用

liyb@lyb:/tmp/test_makefile$ make
cc    -c -o a.o a.c # 由于缺少a.o 但是存在a.c ,于是make使用cc命令自动推导出了a.o的生成命令
g++ a.o -o app  # 这里是makefile的真实的全部的command

2 变量

抛砖引玉:上面说到,make会比较时间戳来判断是否重新生成,那么当依赖非常多的时候,这个操作是非常耗时间的,并且依赖难免有很多重复性的操作,这时候怎么办呢?

三种变量供你使用:

  • 1 自定义变量
# 定义 变量名 = 赋值(必须赋值)
abc = a.c b.c 
# 使用 $(变量名)
gcc $(abc) 
  • 2 预定义变量
    | 变量名 | 含义 | 默认值 |
    | — | — | — |
    | AR | 生成静态库库文件的程序名称 | ar |
    | AS | 汇编编译器的名称 | as |
    | CC | C 语言编译器的名称 | cc |
    | CPP | C 语言预编译器的名称 | $(CC) -E |
    | CXX | C++ 语言编译器的名称 | g++ |
    | FC | FORTRAN 语言编译器的名称 | f77 |
    | RM | 删除文件程序的名称 | rm -f |
    | ARFLAGS | 生成静态库库文件程序的选项 | 无默认值 |
    | ASFLAGS | 汇编语言编译器的编译选项 | 无默认值 |
    | CFLAGS | C 语言编译器的编译选项 | 无默认值 |
    | CPPFLAGS | C 语言预编译的编译选项 | 无默认值 |
    | CXXFLAGS | C++ 语言编译器的编译选项 | 无默认值 |
    | FFLAGS | FORTRAN 语言编译器的编译选项 | 无默认值 |

  • 3 自动变量(只能在规则的命令中使用)

以下是 Makefile 中常用自动变量的含义整理,结合了 Makefile 官方文档和实际使用场景的解析:

变量含义应用场景示例
$*目标文件的主干名(不包含扩展名)在模式规则 %.o: %.c 中,若目标为 main.o,则 $* 表示 main
$+所有依赖文件列表(包含重复项,按出现顺序排列)需要保留重复依赖时使用,例如链接库文件多次依赖的场景。
$<第一个依赖文件的名称编译单个源文件时常用,如 gcc -c $< -o $@
$?比目标文件新的依赖文件列表(以空格分隔)仅重新编译已修改的依赖文件时使用,常用于增量构建优化。
$@目标文件的完整名称(含扩展名)通用规则中指定输出文件名,如 gcc $^ -o $@
$^所有不重复的依赖文件列表(自动去重,以空格分隔)链接多个对象文件时使用,避免重复依赖导致的错误 。

3 模式匹配

使用% 作为通配符

%.0: %.c
	gcc -c $< 

tips:规则同名处理机制

  • 同名规则会合并依赖,后面规则命令会覆盖前面的规则命令
  • 还是遵循只看第一个target的规则,只是叫target的规则不止一个
  • 显示规则会覆盖隐式规则(这时候就不一定第一个生效了)
  • 至于隐式规则,模糊的概念就是make默认的,不需要写出来了,比如上面的自动推导,但这只是其中一种,这里很模糊就不管了。
liyb@lyb:/tmp/test_makefile$ cat makefile 
# 自动推导所有 .c 生成 .o 
%.o: %.c 
	echo "模式匹配 "
app: a.c
	echo "app"
	gcc a.c -o app
liyb@lyb:/tmp/test_makefile$ make
echo "app"
app
gcc a.c -o app


4 函数

所有的函数都是有返回值的,因此需要使用$()解出来

4.1 wildcard

搜索路径下指定类型的文件

# 语法 
wildcard path/*.0   *.c
	*.c 当前目录下.c后缀的文件
# 例子
depends = $(wildcard *.c)
app: $(depends)
	gcc $^ -o app

4.2 patsubst

不改变文件名称替换后缀,这里仅仅是字符串的替换,文件本身没有变化,得到的是一个字符串

src = $(wildcard *.c)
depends = $(patsubst %.c, %.o, $(src))
app: $(depends)
	gcc $^ -o app

5 练习

# 目录结构
.
├── include
│   └── head.h	==> 头文件, 声明了加减乘除四个函数
├── main.c		==> 测试程序, 调用了head.h中的函数
└── src
    ├── add.c	==> 加法运算
    ├── div.c	==> 除法运算
    ├── mult.c  ==> 乘法运算
    └── sub.c   ==> 减法运算

不同方法的理解:

方法1:直接使用.c文件编译

src = $(wildcard ./src/*.c *.c)
# depends = $(patsubst %.c, %.o, $(src)) 
include = ./include
target = app
${target}: $(src) 
	gcc $^  -I$(include) -o $@

# %.o: %.c
#	gcc -c $< -I$(include) -o  $@

# .PHONY:clean

#clean:
#	-rm $(depends) $(target) -f

坏处: 不好设置clean ,因为clean删除的是.0文件,但是这种方法并没有操作.0文件,导致clean的时候还是要查找.o ,那既然如此,不如直接操作.o

方法二:使用.o文件编译

src = $(wildcard ./src/*.c *.c)
depends = $(patsubst %.c, %.o, $(src)) 
include = ./include
target = app
${target}: $(depends) 
	gcc $^ -o $@

%.o: %.c
	gcc -c $< -I$(include) -o  $@

.PHONY:clean
clean:
	-rm $(depends) $(target) -f

  • 先查找所有的源文件
  • 将文件名改名替换得到一个.o的字符串
  • 使用.o作为依赖
  • 模式匹配.o,使用搜索头文件目录的参数-I
  • 因为头文件已经被加入了.o,不需要重复搜索头文件了

5.1 clean

clean作为一个规则定义

  • 由于是伪目标,并且不在第一个规则嵌套里,make并不会执行他
  • .PHONY
    • 用来声明是伪目标
    • 如果不加的话,当目录里真的有这个名称的文件的时候,他就不是伪目标了,他就是一个真的目标,所以就有时间戳,前面说到,如果项目没有更改,就不会执行,那clean就失去了作用
    • 加上.PHONY 声明这是一个伪目标,即使有同名文件,也不影响make把他当做没有时间戳的目标执行
  • 如果不定义clean是没法使用 make clean的,这里明白了clean其实是个规则,并不是默认的一个命令

相关文章:

  • Graphics View画一个可调速的风机(pyqt)
  • 经典算法 最多约数问题
  • MySQL 数据库安全配置最佳实践
  • 【Java】System 类
  • 笛卡尔方法论和解析几何的诞生
  • 【区块链 + 智慧政务】 伽罗华域:区块链数据溯源系统 | FISCO BCOS 应用案例
  • window平台上qtcreator上使用opencv报错
  • 005 公网访问 docker rocketmq
  • 代理服务器与内网穿透/打洞
  • 旁挂组网负载分担组网场景实验
  • HTTP/1.0、HTTP/1.1、HTTP/2 核心区别对比
  • 达梦数据库阻塞死锁及解锁
  • 【SpringBoot】【log】 自定义logback日志配置
  • 算法之刷题汇总
  • 布隆过滤器、布隆算法笔记
  • 线程间的通信
  • [C++] enum 以及 enum class 简单用法
  • Transformer 代码剖析6 - 位置编码 (pytorch实现)
  • Cursor+pycharm接入Codeuim(免费版),Tab自动补全功能平替
  • 支付宝 IoT 设备入门宝典(下)设备经营篇
  • 推荐一些做网站网络公司/2022最新免费的推广引流软件
  • wordpress 有什么用/济南做seo外包
  • 网站设计心的/百度站长统计
  • 赤风设计/seo外链在线工具
  • 网站网页设计html/线下推广有哪几种渠道
  • 超级工程网站建设上海中心大厦/沈阳网站建设