1-7makefile

文章目录

简介

make 是解释 makefile中指令的工具

1 规则

makefile 复制代码
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规则的执行

makefile 复制代码
# in makefile
app: depent1
        echo "first -1"
        echo "first -2"

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

result:

shell 复制代码
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本身,再执行命令

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

  • 执行指定的一条规则

makefile 复制代码
# 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 供目标使用

shell 复制代码
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 自定义变量
makefile 复制代码
# 定义 变量名 = 赋值(必须赋值)
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 模式匹配

使用% 作为通配符

makefile 复制代码
%.0: %.c
	gcc -c $< 

tips:规则同名处理机制

  • 同名规则会合并依赖,后面规则命令会覆盖前面的规则命令
  • 还是遵循只看第一个target的规则,只是叫target的规则不止一个
  • 显示规则会覆盖隐式规则(这时候就不一定第一个生效了)
  • 至于隐式规则,模糊的概念就是make默认的,不需要写出来了,比如上面的自动推导,但这只是其中一种,这里很模糊就不管了。
shell 复制代码
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

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

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

4.2 patsubst

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

makefile 复制代码
src = $(wildcard *.c)
depends = $(patsubst %.c, %.o, $(src))
app: $(depends)
	gcc $^ -o app

5 练习

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

不同方法的理解:

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

makefile 复制代码
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文件编译

makefile 复制代码
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其实是个规则,并不是默认的一个命令
相关推荐
Levin__NLP_CV_AIGC2 小时前
更新 / 安装 Nvidia Driver 驱动 - Ubuntu - 2
linux·运维·ubuntu
DLR-SOFT2 小时前
Windows远程访问Ubuntu的方法
linux·运维·ubuntu
GalenWu2 小时前
对象转换为 JSON 字符串(或反向解析)
前端·javascript·微信小程序·json
GUIQU.2 小时前
【Vue】微前端架构与Vue(qiankun、Micro-App)
前端·vue.js·架构
数据潜水员2 小时前
插槽、生命周期
前端·javascript·vue.js
咸鱼2333号程序员2 小时前
Linux ifconfig命令详解
linux·服务器·网络
秦jh_2 小时前
【Linux网络】应用层协议HTTP
linux·运维·服务器·网络·网络协议·tcp/ip·http
2401_837088502 小时前
CSS vertical-align
前端·html
优雅永不过时·3 小时前
实现一个漂亮的Three.js 扫光地面 圆形贴图扫光
前端·javascript·智慧城市·three.js·贴图·shader
利刃大大3 小时前
【网络编程】四、守护进程实现 && 前后台作业 && 会话与进程组
linux·网络·c++·网络编程·守护进程