makefile编写与使用

makefile简介

  makefile 是一种类似shell的脚本文件,需要make工具进行解释 makefile 内的语句,然后执行内部语句。Makefile的作用是去管理工程项目,比如一个项目有很多c文件,需要利用Makefile去统一进行编译或者其他操作。[1]

  makefile 命名为 makefile 或者 Makefile,如果取名为其他,那么执行时需要使用:

 make -f 自定义makefile文件名字
 
 

1. makefile 基本规则

makefile由一组规则组成,每条规则的格式是:

目标: 依赖
(tab)命令
 
 

其中格式内容的含义是:

​ 目标:想要生成的目标文件

​ 依赖:生成目标文件所需要的文件

​ 命令:通过执行该命令,将依赖文件生成目标文件

2. makefile 中的变量

2.1 自定义变量

  直接使用 = 赋值变量,使用 &(变量名) 来获取变量

name=lisi
person=$(name)
 
 

2.2 自带变量

  makefile 自己默认的一些变量,用户直接赋值。

#CPP编译器名称
CPP=g++
#C编译器名称
CC=gcc
#CPP预处理选项选项 -I 大哎(头文件路径)
CPPFLAGS=-I
#CPP编译器选项
CXXFLAGS=
#C编译器选项 -Wall -g -c
CPPFLAGS=
#链接器选项 -l 小艾尔(链接库名);-L 大艾尔(链接库所在文件夹)
LDFLAGS=
 
 

2.3 自动变量

  makefile 里面一些特定符号代表的特殊含义。

$@ :表示规则中的目标
$< :表示规则中的第一个条件
$^ :表示规则中的所有条件,如果有重复项会消除重复项。
 
 

模式规则

在makefile中 ,% 代表 一个或多个

3. makefile 函数

#1. wildcard  查找指定目录下的指定类型的文件
#查找当前文件夹下所有 .c 文件,赋值给src
#假如当前文件夹下有 fun1.c fun2.c,那么  src=fun1.c fun2.c
src=$(wildcard *.c)
 
 
#2. patsubst  匹配替换,将某个字符串中的某些字符替换为另一个字符
#下面就是将 src 中的 .c 文件替换为 .o 文件,并赋值给 obj
#假如 src=fun1.c fun2.c,那么 obj=fun1.o fun2.o
obj=$(patsubst %.c,%.o,$(src))

#还可以不用函数,使用下面这个
obj=$(src:.c=.o)
也就是把src里面的.c 替换为.o
 
 
#3. addprefix 添加前缀
preobj=$(addprefix objs/, $(obj))
.PHONY:todo
todo:
    @echo $(preobj)
#执行 make todo
#输出 objs/base.o
 
 
#4. filter 过滤器
files = $(wildcard *.*)
files := $(filter %.cpp %.c, $(files))
.PHONY:filter
filter:
    @echo $(files)
#执行 make filter
#输出 base.c base.cpp
 
 
#5. filter-out 反过滤器,和上面的相反,这个是去除匹配到的
files = $(wildcard *.*)
files := $(filter-out%.cpp %.c, $(files))
.PHONY:filter
filter:
    @echo $(files)
#执行 make filter
#输出 base.h base.hpp
 
 
#6. strip 去除字符串间多余空格
strips = a.c  b.c       d.c
strips :=$(strip $(strips))
.PHONY:strip
strip:
    @echo $(strips)
#执行 make strip
#输出 a.c b.c d.c
 
 
#7. subst 将字符串substs中的所有 .c 替换为 .o
substs = .c.c.o .o.c.c
substs := $(subst .c,.o,$(substs))
.PHONY:substs
substs:
    @echo $(substs)
#执行 make substs
#输出 .o.o.o .o.o.o
 
 
#8. abspath 将相对路径转换为绝对路径
RELATIVE_PATH = ./base.c 
ABSOLUTE_PATH = $(abspath $(RELATIVE_PATH)) 
.PHONY:abspath
abspath: 
    @echo "Relative path: $(RELATIVE_PATH)" 
    @echo "Absolute path: $(ABSOLUTE_PATH)"
#执行 
make abspath
#输出 
Relative path: ./base.c 
Absolute path: /home/user/base.c
 
 
#9. word 系列函数
word=$(word 3, a b c)    #获取字符串中第3个单词
words=$(words a b c) #返回有多少个单词
firstword=$(firstword a b c)
lastword=$(lastword a b c)
curdir = $(shell pwd)
.PHONY:wordsys
wordsys:
    @echo "3 word is $(word)"
    @echo "total word is $(words)"
    @echo "first word is $(firstword)"
    @echo "lastword word is $(lastword)"

#执行
make wordsys
#输出
3 word is c
total word is 3 
first word is a
lastword word is c
 
 
#10. shell 执行shell命令,并将结果赋值给变量
curdir = $(shell ls)
.PHONY:lldir
lldir:
    @echo "curdir is $(curdir)"

#执行
make lldir
#输出 文件列表
 
 
#11. 日志打印函数
$(info "infoing...")
$(warning "warning...")
$(error "erroring...")

#在解析Makefile 的时候会输出,如果是error会暂停执行
#输出
"infoing..."
Makefile:3: "warning..."
Makefile:4: *** "erroring...".  Stop.
 
 

4. 伪目标

  伪目标的作用是为了让 makefile 不去检查当前目标是否为最新,不管是不是最新都执行这个目标的命令。

#格式
# 伪目标
.PHONY:clean
clean:
        rm -rf $(target) $(obj)
 
 

  需要伪目标的原因就是,make 在解析 makefile 时,它首先去执行第一条规则,这条规则称为终极规则。假如这条规则里面的 依赖有更新,那么就会去找依赖对应的规则去执行,然后再执行最终规则。假如依赖都是最新,那么不会执行最终规则。可以看到如下提示:

#执行
make
#输出
make: 'xxx' is up to date.
# 表示当前最终规则是最新的,不需要执行最终规则

####假如如果当前目录下有同名clean文件####
#一定要添加 .PHONY 表示 clean 目标是一个伪目标,表示不需要检查文件是否存在或者已更新,会直接去执行这个伪目标的命令
#如果不添加,由于clean 这个目标的规则没有依赖,所以认为它的依赖是最新的,然后检查是否存在clean文件,发现确实存在一个没有更新的名为 clean 的文件,最终导致clean不会执行.
#例如直接写下面这样,不添加 .PHONY , 并且存在一个名为 clean 的文件:
clean:
        rm -rf $(target) $(obj)

#执行
make clean
#输出以下内容,无法执行目标命令。
make: 'clean' is up to date.
 
 

5. 特殊变量

5.1 MAKECMDGOALS 变量

#全局变量
.PHONY:show args
show args:
    @echo "\$$@ = $@"
    @echo "MAKECMDGOALS = $(MAKECMDGOALS)"

#执行
make show
#输出
$@ = show
MAKECMDGOALS = show

#执行
make show args
#输出
$@ = show
MAKECMDGOALS = show args
$@ = args
MAKECMDGOALS = show args
 
 

5.2 MAKEFILE_LIST 变量

获取包含的 Makefile 列表

name1 := $(word $(words $(MAKEFILE_LIST)),$(MAKEFILE_LIST))
include tttfile
name2 := $(word $(words $(MAKEFILE_LIST)),$(MAKEFILE_LIST))
show:
    @echo $(name1)
    @echo $(name2)
#执行
make -f mmmfile
#输出
mmmfile
tttfile
 
 

6. include 命令

包含另一个Makefile 文件,让另一个 Makefile 文件 在本文件中展开,另一个文件的变量能够在本文件内直接使用

# mmmfile 文件
CFLAGS= abc
#Makefile 文件
include mmmfile
.PHONY:all
all:
    @echo $(CFLAGS)
#执行
make all
#输出
abc
 
 

7. export 命令

可以在Makefile中导出 全局变量 ,并用在shell环境中

export nums=10
.PHONY:expnum
expnum:
    @echo $(nums)
#执行
make expnum
#输出
10
 
 

8. 获取make 后附带的变量值

.PHONY:expval
expval:
    @echo $(a)
#执行 make expnum a='showtime'
#输出 showtime
 
 

基础综合案例

用到四个基础文件 file1.c、file2.c 、head.h、hello.c,一个 makefile 文件

其中文件file1.c 如下:

#include"head.h"

int sum(int a, int b)
{
   
        return a+b;
}
 
 

其中文件file2.c 如下:

#include"head.h"

int mul(int a,int b)
{
   
        return a*b;
}
 
 

其中文件head.h如下:

#include<stdio.h>
int sum(int a,int b);
int mul(int a,int b);
 
 

其中调用文件 hello.c 如下:

#include"head.h"

int main()
{
   
        printf("1+2=%d\n",sum(1,2));
        printf("2*2=%d\n",mul(2,2));
        return 0;
}
 
 

makefile文件:

src=$(wildcard ./*.c)
obj=$(patsubst %.c,%.o,$(src))
target=hello
CC=gcc
CPPFLAGS=-I ./
# 最终目标
$(target):$(obj)
        $(CC) -o $@ $^
%.o:%.c
        $(CC) -o $@ -c $< $(CPPFLAGS)

# 伪目标
.PHONY:clean
clean:
        rm -rf $(target) $(obj)
 
 

执行

#执行,它会自动去执行终极目标的命令 $(CC) -o $@ $^
make
#输出以下表示成功
gcc -o file2.o -c file2.c -I ./
gcc -o hello.o -c hello.c -I ./
gcc -o file1.o -c file1.c -I ./
gcc -o hello file2.o hello.o file1.o
#执行可执行文件
./hello
#输出以下,成功
1+2=3
2*2=4

#指定执行清除指令
make clean
#输出以下表示成功
rm -rf hello ./file2.o ./hello.o ./file1.o
posted @ 2025-10-28 11:35  风一直那个吹  阅读(29)  评论(0)    收藏  举报