Makefile 学习

2019-02-18  本文已影响0人  O2Space_Xiu

Makefile

一、Makefile 简单使用

Makefile是Linux系统下的一种编译脚本,更快、更方便的编译一个c/c++程序


二、简单了解编译和链接过程

源代码(helloworld.c)->预处理器->编译器->汇编代码(helloworld.s)->汇编器(helloworld.o)->目标代码->链接器->可执行程序(helloworld.exe)

冯·诺依曼结构:CPU、硬盘(断电不消失)、内存(断电消失)
启动运行过程 从硬盘镜像代码加载到内存中,cpu一条条去读取加载到内存中的指令去执行


三、Makefile基本语法

规则、变量、条件执行、文本/文件名处理函数、文件包含、注释

规则

target: dependencies
    command

注释

#

变量

cc = gcc

条件

ifeq ($(DEBUG), "true")
cc = gcc -g
else
cc = gcc
endif

搜索当前 目录下的所有.c文件

SRCS = $(willdcard *.c)

加前缀 目录加到文件名前面

OBJS = $(addprefix 目录,文件名)

文件包含

# 类似于C语言的#include,这里使用include命令
include makefile内容的文件

函数

依赖关系树
Makefile的目的:构建依赖关系树

如何表示依赖关系树

依赖关系树的生命周期


四、规则基本构成

目标:目标依赖
    命令

4.1.注意事项

test:
    @echo "test..."
clean:
    rm *.o
all:test

4.2.目标

除伪目标外每个目标都会生成目标文件

targetX targetY:
    命令
all: targetX
all: targetY
#等同
all: targetX targetY

targetX:
    @echo ""
targetY:
    @echo ""
.PHONY:all clean
all: targetX targetY
targetX:
    @echo ""
targetY:
    @echo ""
clean:
    rm *.o

4.3.目标依赖

gcc -M main.c
gcc -MM main.c
cc = gcc
main:foo.o bar.o
    $(cc) -o main foo.o bar.o
#a.o:a.c
#   $(cc) -c foo.c
#b.o:b.c
#   $(cc) -c bar.c
# $@ = 目标文件,$^ = 所有的依赖文件,$< = 第一个依赖文件。
cc = gcc
main:foo.o bar.o
    $(cc) -o main foo.o bar.o
%.o:%c
    $(cc) -o $@ -c $^    

4.4.生成命令

cc = gcc
main:foo.o ba.o     
    @echo "start build main"
    $(cc) -o main foo.o bar.o
    @echo "build main success"
%.o:%c
    $(cc) -o $@ -c $^
make -j3 #开启多个进程执行
all:
    cd /
    pwd     #显示当前目录
#改成下面方式,同一个进程 命令间使用;(分号)+ (空格)+\(分隔符),变成一个逻辑命令
all:
    cd /; \
    pwd


五、变量

5.1.变量基础

5.1.1.变量定义
5.1.2.变量赋值
STR = hello
STR += world!

v1 = a
v1 ?= b
v2 ?= b

all:
    @echo "STR = $(STR)" # hello world!
    @echo "v1 = $(v1)"   # a
    @echo "v2 = $(v2)"   # b

5.1.3.变量引用
cc = gcc
BIN = main
OBJS = foo.o bar.o
$(BIN):$(OBJS)
    $(cc) -o $(BIN) $(OBJS)
%.o:%.c
    $(cc) -o $@ -c $^

5.2.变量分类

5.2.1.立即展开变量
.PHONY:all

HELLO = Good
TIME = morning!
STRING := $(HELLO) $(TIME)
$(info $(STRING))
TIME = afternoon!
$(info $(STRING))

all:
    @echo "done" 
    
# 结果
# Good morning!
# Good morning!
# done
5.2.2.延迟展开变量
.PHONY:all

HELLO = Good
TIME = morning!
STRING = $(HELLO) $(TIME)
$(info $(STRING))
TIME = afternoon!
$(info $(STRING))

all:
    @echo "done" 
    
# 结果
# Good morning!
# Good afternoon!
# done
5.2.3.注意事项

5.3.目标变量

5.3.1.一般变量
5.3.2.目标变量
cc = gcc
BIN = main
OBJS = foo.o bar.o
N = 1
$(BIN): N = 2
$(BIN):$(OBJS)
    @echo "BIN: N = $(N)"   # 2
    $(cc) -o $(BIN) $(OBJS)
foo.o: N = 3
foo.o:foo.c
   @echo "foo.o: N = $(N)" # 3
    $(cc) -c foo.c
bar.o:bar.c
   @echo "bar.o: N = $(N)" # 这里是几呢?
    $(cc) -c bar.c

clean:
    @echo "clean: N = $(N)" # 1

5.3.3.使用目标变量

5.4.模式变量

5.4.1目标变量
5.4.2.模式变量
cc = gcc
BIN = main
OBJS = foo.o bar.o
N = 1
$(BIN): N = 2
$(BIN):$(OBJS)
    @echo "BIN: N = $(N)"   # 2
    $(cc) -o $(BIN) $(OBJS)
%.o: N = 3
foo.o:foo.c
   @echo "foo.o: N = $(N)" # 3
    $(cc) -c foo.c
bar.o:bar.c
   @echo "bar.o: N = $(N)" # 3
    $(cc) -c bar.c

clean:
    @echo "clean: N = $(N)" # 1

5.5.自动变量

5.5.1自动变量是局部变量
5.5.2目标
5.5.3所有依赖
5.5.4.第一个依赖
5.5.5.使用举例
cc = gcc
BIN = main
OBJS = foo.o bar.o
$(BIN):$(OBJS)
    @echo "BIN----------$@:$^"
    $(cc) -o $@ $^
foo.o:foo.c
    @echo "foo----------$@:$^"
    $(cc) -o $@ -c $^
bar.o:bar.c
    @echo "bar----------$@:$^"
    $(cc) -o $@ -c $^

5.6.系统环境变量

5.6.1.作用范围
5.6.2.常见的系统环境变量
.PHONY:all

all:
    @echo "CFLAGS = $(CFLAGS)"
    @echo "SHELL = $(SHELL)"          # /bin/sh
    @echo "MAKE = $(MAKE)"            # make
    @echo "HOSTNAME = $(HOSTNAME)"

5.7.变量的传递

5.7.1.Makefile在多目录下递归执行
N = 3
all:
    @echo "build...."
    cd test && make N = $(N)
5.7.2.通过export传递变量
export N = 3
all:
    @echo "build...."
    cd test && make
5.7.3.通过命令行传递变量


六、条件执行

常用形式

其它合法形式

上面‘ifxxx’为下方关键字

关键字 功能
ifeq 判断参数是否相等,相等为true,否则为false
ifneq 判读参数是否不相等,不相等为true,否则为false
ifdef 判断变量是否有值,有值为true,否则为false
ifndef 判断变量是否没有值,没有值为true,否则为false
# Makefile 内容
x := A
y := $(x)
z :=

test:
ifeq ($(x),$(y)) # 注意:在ifeq 前面不能使用\tab键,而是使用空格键
    @echo "x == y"
else
    @echo "x != y"
endif

ifneq ($(x),$(y))
    @echo "x != y"
else
    @echo "x == y"
endif
    
 ifdef $(y)
    @echo "y is Not empty"
 else
    @echo "y is empty"
 endif
 
 ifndef $(z)
    @echo "z is empty"
 else
    @echo "z is Not empty"
 endif
   
# bash 中执行 make
$make
##输出:
x == y
x == y
y is Not empty
z is empty

多分支格式:

x = 123
y = 456
z = 123
ifeq $(x),$(y)
    @echo "x == y"
else ifeq $(x),$(z)
    @echo "x == z"
else
    @echo "x != y && x != z"
endif


七、函数

Makefile中自带了一些函数,利用这些函数可以简化Makefile的编写
函数调用语法如下:

$(<function> <arguments>)
#或者
${<function> <arguments>}

7.1.字符串函数

7.1.1.字符串替换函数:$(subst <from>,<to>,<text>)*

功能:把字符串<text>中的<from>替换为<to>
返回:替换过的字符串

# Makefile 内容
all:
    @echo $(subst t,e,maktfilt)
    
# bash 中执行 make
$make
##输出:
makefile
7.1.2.模式字符串替换函数: $(patsubst <pattern>,<replacement>,<text>)

功能: 查找<text>中的单词(单词以"空格", "tab", "换行"来分割) 是否符合 <pattern>, 符合的话, 用 <replacement> 替代.
返回: 替换过的字符串

# Makefile 内容
all:
    @echo $(patsubst %.c,%.o,foo.c bar.c)

# bash 中执行 make
$make
##输出: 
foo.o bar.o
7.1.3.去空格函数: $(strip <string>)

功能: 去掉 <string> 字符串中开头和结尾的空字符
返回: 被去掉空格的字符串值

# Makefile 内容
VAL := "      foo.c bar.c foo.o bar.o "
all:
    @echo "去除空格前:" $(VAL)
    @echo "去除空格后:" $(strip $(VAL))

# bash 中执行 make
$make
##输出:
      foo.c bar.c foo.o bar.o 
foo.c bar.c foo.o bar.o
7.1.4.查找字符串函数: $(findstring <find>,<in>)

功能: 在字符串 <in> 中查找 <find> 字符串
返回: 如果找到, 返回 <find> 字符串, 否则返回空字符串

# Makefile 内容
VAL := "      foo.c bar.c foo.o bar.o "
all:
    @echo $(findstring foo,$(VAL))
    @echo $(findstring fbar,$(VAL))

# bash 中执行 make
$make
##输出:
foo 

7.1.5.过滤函数: $(filter <pattern...>,<text>)

功能: 以 <pattern> 模式过滤字符串 <text>, 保留 符合模式 <pattern> 的单词, 可以有多个模式
返回: 符合模式 <pattern> 的字符串

# Makefile 内容
all:
    @echo $(filter %.o %.a,foo.c foo.o main.a)


# bash 中执行 make
$ make
##输出:
foo.o main.a
7.1.6.反过滤函数: $(filter-out <pattern...>,<text>)

功能: 以 <pattern> 模式过滤字符串 <text>, 去除 符合模式 <pattern> 的单词, 可以有多个模式
返回: 不符合模式 <pattern> 的字符串

# Makefile 内容
all:
    @echo $(filter-out %.o %.a,foo.c foo.o main.a)


# bash 中执行 make
$ make
##输出:
foo.c
7.1.7.排序函数: $(sort <list>)

功能: 给字符串 <list> 中的单词排序 (升序)
返回: 排序后的字符串

# Makefile 内容
all:
    @echo $(sort bac abc acb cab)

# bash 中执行 make
$ make
##输出:
abc acb bac cab
7.1.8.取单词函数: $(word <n>,<text>)

功能: 取字符串 <text> 中的 第<n>个单词 (n从1开始)
返回: <text> 中的第<n>个单词, 如果<n> 比 <text> 中单词个数要大, 则返回空字符串

# Makefile 内容
all:
    @echo $(word 1,aa bb cc dd)
    @echo $(word 5,aa bb cc dd)
    @echo $(word 4,aa bb cc dd)

# bash 中执行 make
$ make
##输出:
aa

dd
7.1.9.取单词串函数: $(wordlist <s>,<e>,<text>)

功能: 从字符串<text>中取从<s>开始到<e>的单词串. <s>和<e>是一个数字.
返回: 从<s>到<e>的字符串

# Makefile 内容
all:
    @echo $(wordlist 1,3,aa bb cc dd)
    @echo $(word 5,6,aa bb cc dd)
    @echo $(word 2,5,aa bb cc dd)


# bash 中执行 make
$ make
##输出:
aa bb cc

bb
7.1.10.单词个数统计函数: $(words <text>)

功能: 统计字符串 <text> 中单词的个数
返回: 单词个数

# Makefile 内容

all:
    @echo $(words aa bb cc dd)
    @echo $(words aabbccdd)
    @echo $(words )

# bash 中执行 make
$ make
##输出:
4
1
0
7.1.11.首单词函数: $(firstword <text>)

功能: 取字符串 <text> 中的第一个单词
返回: 字符串 <text> 中的第一个单词

# Makefile 内容
all:
    @echo $(firstword aa bb cc dd)
    @echo $(firstword aabbccdd)
    @echo $(firstword )

# bash 中执行 make
$ make
##输出:
aa
aabbccdd

7.2.文件名函数

7.2.1.取目录函数: $(dir <names...>)

功能: 从文件名序列 <names> 中取出目录部分
返回: 文件名序列 <names> 中的目录部分

# Makefile 内容
all:
    @echo $(dir /home/a.c ./bb.c ../c.c d.c)

# bash 中执行 make
$ make
##输出:
/home/ ./ ../ ./
7.2.2.取文件函数: $(notdir <names...>)

功能: 从文件名序列 <names> 中取出非目录部分
返回: 文件名序列 <names> 中的非目录部分

# Makefile 内容
all:
    @echo $(notdir /home/a.c ./bb.c ../c.c d.c)

# bash 中执行 make
$ make
##输出:
a.c bb.c c.c d.c
7.2.3.取后缀函数: $(suffix <names...>)

功能: 从文件名序列 <names> 中取出各个文件名的后缀
返回: 文件名序列 <names> 中各个文件名的后缀, 没有后缀则返回空字符串

# Makefile 内容
all:
   @echo $(suffix /home/a.c ./b.o ../c.a d)

# bash 中执行 make
$ make
##输出:
.c .o .a
7.2.4.取前缀函数: $(basename <names...>)

功能: 从文件名序列 <names> 中取出各个文件名的前缀
返回: 文件名序列 <names> 中各个文件名的前缀, 没有前缀则返回空字符串

# Makefile 内容
all:
    @echo $(basename /home/a.c ./b.o ../c.a /home/.d .e)

# bash 中执行 make
$ make
##输出:
/home/a ./b ../c /home/
7.2.5.加后缀函数: $(addsuffix <suffix>,<names...>)

功能: 把后缀 <suffix> 加到 <names> 中的每个单词后面
返回: 加过后缀的文件名序列

# Makefile 内容
all:
    @echo $(addsuffix .c,/home/a b ./c.o ../d.c)

# bash 中执行 make
$ make
##输出:
/home/a.c b.c ./c.o.c ../d.c.c
7.2.6.加前缀函数: $(addprefix <prefix>,<names...>)

功能: 把前缀 <prefix> 加到 <names> 中的每个单词前面
返回: 加过前缀的文件名序列

# Makefile 内容
all:
    @echo $(addprefix test_,/home/a.c b.c ./d.c)

# bash 中执行 make
$ make
##输出:
test_/home/a.c test_b.c test_./d.c
7.2.7.连接函数: $(join <list1>,<list2>)

功能: <list2> 中对应的单词加到 <list1> 后面
返回: 连接后的字符串

# Makefile 内容
all:
    @echo $(join a b c d,1 2 3 4)
    @echo $(join a b c d,1 2 3 4 5)
    @echo $(join a b c d e,1 2 3 4)

# bash 中执行 make
$ make
##输出:
a1 b2 c3 d4
a1 b2 c3 d4 5
a1 b2 c3 d4 e

7.3.foreach

语法:
$(foreach <var>,<list>,<text>)

# Makefile 内容
targets := a b c d
objects := $(foreach i,$(targets),$(i).o)

all:
    @echo $(targets)
    @echo $(objects)

# bash 中执行 make
$ make
##输出:
a b c d
a.o b.o c.o d.o

7.4.if

这里的if是个函数, 和前面的条件判断不一样, 前面的条件判断属于Makefile的关键字
语法:
$(if <condition>,<then-part>)

$(if <condition>,<then-part>,<else-part>)

# Makefile 内容
val := a
objects := $(if $(val),$(val).o,nothing)
no-objects := $(if $(no-val),$(val).o,nothing)

all:
    @echo $(objects)
    @echo $(no-objects)

# bash 中执行 make
$ make
##输出:
a.o
nothing

7.5.call - 创建新的参数化函数

语法:
$(call <expression>,<parm1>,<parm2>,<parm3>...)

# Makefile 内容
log = "====debug====" $(1) "====end===="

all:
    @echo $(call log,"正在 Make")

# bash 中执行 make
$ make
##输出:
====debug==== 正在 Make ====end====

7.6.origin - 判断变量的来源

语法:
$(origin <variable>)
返回值有如下类型:

类型 含义
undefined <variable> 没有定义过
default <variable> 是个默认的定义, 比如 CC 变量
environment <variable> 是个环境变量, 并且 make时没有使用 -e 参数
file <variable> 定义在Makefile中
command line <variable> 定义在命令行中
override <variable> 被 override 重新定义过
automatic <variable> 是自动化变量
# Makefile 内容
val-in-file := test-file
override val-override := test-override

all:
    @echo $(origin not-define)    # not-define 没有定义
    @echo $(origin CC)            # CC 是Makefile默认定义的变量
    @echo $(origin PATH)         # PATH 是 bash 环境变量
    @echo $(origin val-in-file)    # 此Makefile中定义的变量
    @echo $(origin val-in-cmd)    # 这个变量会加在 make 的参数中
    @echo $(origin val-override) # 此Makefile中定义的override变量
    @echo $(origin @)             # 自动变量, 具体前面的介绍

# bash 中执行 make
$ make val-in-cmd=val-cmd
##输出:
undefined
default
environment
file
command line
override
automatic

7.7.shell

语法:
$(shell <shell command>)
它的作用就是执行一个shell命令, 并将shell命令的结果作为函数的返回.
作用和 <shell command> 一样

7.8.make 控制函数

7.8.1.产生一个致命错误: $(error <text ...>)

功能: 输出错误信息, 停止Makefile的运行

# Makefile 内容
all:
    $(error there is an error!)
    @echo "这里不会执行!"

# bash 中执行 make
$ make
##输出:
Makefile:2: *** there is an error!.  Stop.
7.8.2.输出警告: $(warning <text ...>)

功能: 输出警告信息, Makefile继续运行

# Makefile 内容
all:
    $(warning there is an warning!)
    @echo "这里会执行!"

# bash 中执行 make
$ make
##输出:
Makefile:2: there is an warning!
这里会执行!

八、Makefile中一些GNU约定俗成的伪目标

如果有过在Linux上, 从源码安装软件的经历的话, 就会对 make clean, make install 比较熟悉.

举例:

伪目标 含义
call 所有目标的目标,其功能一般是编译所有的目标
clean 删除所有被make创建的文件
install 安装已编译好的程序,其实就是把目标可执行文件拷贝到指定的目录中去
print 列出改变过的源文件
tar 把源程序打包备份. 也就是一个tar文件
dist 创建一个压缩文件, 一般是把tar文件压成Z文件. 或是gz文件
TAGS 更新所有的目标, 以备完整地重编译使用
check 或 test 一般用来测试makefile的流程
上一篇 下一篇

猜你喜欢

热点阅读