1. make

  • 编译:将源代码文件翻译成处理器可执行的二进制文件的过程,这个过程的时间区间称为编译时
  • 构建:指定多个编译过程的先后顺序

make命令是常用的构建工具,诞生于1977年,主要用于C/C++项目工程的构建,make命令的适用场景:只要某一个或多个文件发生变动,就要重新构建项目

2. Makefile

make的语义是制作,制作出某样工件,工件可以是各种可执行文件、中间文件、镜像等,通过Makefile文件描叙构建工件的过程,构建过程由一个或多个规则组成

  • make 命令默认使用工程项目目录下的Makefile文件,可以通过 -f --file 选项指定其位置的Makefile文件

  • 规则(rules)

<target>: <precondition> ...
|table|<command>
...
规则文法:
1. 冒号前面的 <target> 是构建的目标,可以是所构建对象的文件名,或者构建过程达成的目标名
2. 冒号后面的 <precondition> ... 是前置条件,判断目标是否重新构建,重新构建的条件为:只要有一个前置文件不存在或者有过更新(前置文件的mtime比目标的mtime新),跳过已经编译但未修改的文件,降低总编译时间
3. |table| 是一个缩进,必须以该字符起首
4. <command> 是命令,达成该目标需要执行的子命令
5. |table| <command> 可以有多个,是具体的执行过程,<prerequest> |table|<command> 至少存在一个
2.1 目标 -- target

一个目标构成一条规则

  • 目标可以是文件名
# 会在目录下创建 a.txt b.txt文件,b.txt文件先于a.txt文件创建,若b.txt文件的mtime时间戳比a.txt文件的时间戳新则重新执行a.txt的构建过程
# 通过判断当前工程目录下的文件是否存在,判断是否重新执行目标的构建过程
a.txt: b.txt
cat b.txt > a.txt b.txt:
echo "make comand" > b.txt
  • 目标可以是某个操作名,若工程目录存在与操作同名的文件,就会跳过该目标的执行过程,可以通过 .PHONY声明为伪目标,不会检查目标名对应的文件是否存在
# 若当前工程目录下已存在clean文件,则跳过,可以通过 .PHONY指定该目标是一个非文件目标
.PHONY: clean
clean:
rm a.txt b.txt
2.2 前置条件 -- precondition

一组以空格隔开的文件名,用于判断目标是否重新构建,如果至少有一个文件不存在或者文件的最后修改时间比目标文件的最后修改时间新,则重新构建该目录

# 会首先依次构建前置条件中的目标文件,最后构建目标文件,反复构建,只要前置条件中的文件最后修改时间戳比目标文件小则跳过目标文件的构建过程
result.txt: file1.txt file2.txt file3.txt
cat file1.txt file2.txt file3.txt > result.txt file1.txt:
echo "file1" > file1.txt file2.txt:
echo "file2" > file2.txt file3.txt:
echo "file3" > file3.txt
2. 3 命令 -- command

描叙目标是如何创建出来的,由一个或多个shell命令描叙,是构建目标的具体指令,每一行命令在一个进程中并行执行

  • 每个构建命令在各自的shell进程中执行,通过 .ONESHELL指定目标的命令集合都在同一个shell进程中串行
# 看不到变量NAME的值,需要使用$进行$符号的转义,输出不了变量的值
var-get:
export NAME=cloud
echo $$NAME
# 写在一行的方式,以分号隔开shell指令
var-get:
export NAME=cloud;echo $$NAME
# export 与 echo 指令都在同一个shell进程中运行
.ONESHELL:
var-get:
export NAME=cloud
echo $$NAME
  • 可以通过 .RECIPEPREFIX 修改命令的前缀,默认为Table字符
result.txt: file1.txt file2.txt file3.txt
cat file1.txt file2.txt file3.txt > result.txt # 之后的构建目标的指令列表由 > 开头
.RECIPEPREFIX = >
file1.txt:
> echo "file1" > file1.txt file2.txt:
> echo "file2" > file2.txt file3.txt:
> echo "file3" > file3.txt

3. 文法

向make命令描叙字符文字的语法与语义

3.1 注释

以 # 开头,到行尾结束,Bash 与 Python语言 的注释也一样

# export 与 echo 指令都在同一个shell进程中运行
.ONESHELL:
var-get:
export NAME=cloud # 设置一个环境变量
echo $$NAME # 输出环境变量的值
3. 2 回声

echo -- 执行make命令会打印出每个目标的命令然后再执行,在命令的最开头添加@关闭回声

# export 与 echo 指令都在同一个shell进程中运行
.ONESHELL:
var-get:
@export NAME=cloud
echo $$NAME
3.2 通配符

选定一组文件名符合通配符的文件,文法规则和bash一致

  • 单个字符
  • 任意长度任意字符*
  • 字符集中单个字符 []
# 删除所有以 .0 结尾的文件
clean:
rm -f *.o
3.3 模式匹配

选定文件名相同,但文件类型不同的文件

# 所有 .c 结尾的文件通过命令生成同名的.o文件
%.o: %.c
3. 4 变量与赋值符

声明变量并存储值,在之后代码中访问并引用变量中值

  • 赋值 = := ?= +=
VARIABLE = value
# 在执行时扩展,允许递归扩展。 VARIABLE := value
# 在定义时扩展。 VARIABLE ?= value
# 只有在该变量为空时才设置值。 VARIABLE += value
# 将值追加到变量的尾端。
  • 访问值

通过 $()访问,访问bash变量需要加上$进行转义

# 访问定义的变量与bash变量
name = dream_flish
get-variable:
@echo $(name)
@echo $$HOME
  • 内置变量

当前编辑器 CC ,当前使用的make工具 $(MAKE)

# 访问定义的变量与bash变量
name = dream_flish
get-variable:
@echo "$(CC) $(MAKE)"
@echo $$HOME
  • 自动变量

变量的值与当前的规则的目标与前置条件相关

1. $@ 当前目标
2. $< 第一个前置条件
3. $? 比目标文件mtime时间戳更新的前置条件文件,多个以空格隔开
4. $^ 所有前置条件
5. $* 匹配 % 部分
6. $(@D) 目标当前目录名
7. $(@F) 目标当前文件名
8. $(<D) 第一个前置条件的目录名
9. $(<F) 第一个前置条件的文件名
a.txt: b.txt c.txt
echo $@ # 等价 echo a.txt
cat $^ > a.txt # 等价 cat b.txt c.txt > a.txt b.txt:
echo "b" > b.txt c.txt:
echo "c" > c.txt
3. 5 判断与循环

流程控制,满足条件时执行一段代码,不满足时执行另一段代码,利用Bash代码实现循环

BOY_NAMES = ALEX BEI XIXI
GIRL_NAMES = LANGLANG XUEXUE XIAOXIAO
CONDATION = BOY # 等值判断
ifeq ($(CONDATION), BOY)
NAMES=$(BOY_NAMES)
else
NAMES=$(GIRL_NAMES)
endif # 访问变量中的值
all:
for name in $(NAMES); do echo $$name; done

4. 函数

完成某项功能的指令集合,调用文法: $(函数名, 参数 ... )

  • shell函数 -- 执行shell命令
files := $(shell seq 1 10)

all:
echo $(files)
  • wildcard 通配符函数 -- 替换Bash的通配符
# 列出 src目录下所有以.tst后缀结尾的文件
files := $(wildcard src/*.txt)
all:
echo $(files)
  • subst函数 -- 文本替换
# 小写的i换成大写的I
result := $(subst i,I,i Love C++)
all:
echo $(result)

5. 编译 C++代码

  1. 在工程目录创建src目录,再在src目录下创建main.cpp文件,写入以下内容
#include <iostream>

int main() {
std::cout << "I Love C++\n";
}
  1. 在工程目录下创建Makefile文件,写入以下规则
TARGET_DIR = bin
all:
[ ! -d $)TARGET_DIR ] && mkdir -p $(TARGET_DIR); \
c++ src/main.cpp -o $(TARGET_DIR)/main clean:
rm -f bin/*
  1. 在工程目录执行构建命令并执行编译后的结果
# 编译,没有目标则默认找到的第一个目标
make # 执行,会输出 I Love C++
./bin/main # 清理,make命令中输出目标名,多个目标以空格隔开
make clean

makefile文件详解的更多相关文章

  1. MakeFile 文件详解

    GNU的make工作时的执行步骤入下:(想来其它的make也是类似)      1.读入所有的Makefile.      2.读入被include的其它Makefile.      3.初始化文件中 ...

  2. Android.mk文件详解(转)

    源:Android.mk文件详解 从对Makefile一无所知开始,折腾了一个多星期,终于对Android.mk有了一个全面些的了解.了解了标准的Makefile后,发现Android.mk其实是把真 ...

  3. Android.mk文件和Application.mk文件详解

    Android.mk文件和Application.mk文件详解 相同点:都是轻量级 Makefile(Makefile文件定义一些列指令,指定文件编译顺序,类似shell脚本) Application ...

  4. web.xml文件详解

      web.xml文件详解 Table of Contents 1 listener. filter.servlet 加载顺序 2 web.xml文件详解 3 相应元素配置 1 listener. f ...

  5. Linux中/proc目录下文件详解

    转载于:http://blog.chinaunix.net/uid-10449864-id-2956854.html Linux中/proc目录下文件详解(一)/proc文件系统下的多种文件提供的系统 ...

  6. SUBLIME TEXT 2 设置文件详解

    SUBLIME TEXT 2 设置文件详解 Preferences.sublime-settings文件: // While you can edit this file, it’s best to ...

  7. [转]AndroidManifest.xml文件详解

    转自:http://www.cnblogs.com/greatverve/archive/2012/05/08/AndroidManifest-xml.html AndroidManifest.xml ...

  8. delphi 资源文件详解

    delphi资源文件详解 一.引子: 现在的Windows应用程序几乎都使用图标.图片.光标.声音等,我们称它们为资源(Resource).最简单的使用资源的办法是把这些资源的源文件打入软件包,以方便 ...

  9. VSFTPD全攻略(/etc/vsftpd/vsftpd.conf文件详解)

    /etc/vsftpd/vsftpd.conf文件详解,分好类,方便大家查找与学习 #################匿名权限控制############### anonymous_enable=YE ...

  10. jni.h头文件详解二

    作者:左少华 博客:http://blog.csdn.net/shaohuazuo/article/details/42932813 转载请注明出处:http://blog.csdn.net/shao ...

随机推荐

  1. EasyDO这么好用!别家的运维小哥都馋哭了

    运维这份工作似乎总是被人误会- 修服务器的 接网线的 盯监控的 维护系统的 说啥的都有,平常人都不太知道运维是干啥的 跟小编来探一探云掣MSP服务线运维小哥的实际运维工作? 运维小哥一部分日常工作就是 ...

  2. JavaFx 打包jar(六)

    JavaFx 打包jar(六) JavaFX 从入门入门到入土系列 我们编写了不少javafx,那么如何打包成jar给用户呢?下面我给出比较全的打包方式. 打包jar 下面我给出比较全的打包方式. 1 ...

  3. 如何将没有复制或移动构造函数的对象放入vector容器

    正文 直接说答案,这个问题无法实现.原因是因为std::vector容器的插入一定会调用类对象的构造函数或者移动构造函数. 说一下为什么会有这个问题,因为不想用指针,我想直接通过类对象本身的RAII机 ...

  4. 昇腾实践丨ATC模型转换动态shape问题案例

    本文分享自华为云社区<ATC模型转换动态shape问题案例>,作者:昇腾CANN. ATC(Ascend Tensor Compiler)是异构计算架构CANN体系下的模型转换工具:它可以 ...

  5. 华为云GaussDB两大数据库通过中国信通院多项评测

    摘要:权威认可!华为云GaussDB两大数据库通过中国信通院多项评测. 本文分享自华为云社区<官宣!华为云GaussDB两大数据库通过中国信通院多项评测>,作者: GaussDB 数据库 ...

  6. 教你用Java7的Fork/Join框架开发高并发程序

    摘要:Fork/Join框架位于J.U.C(java.util.concurrent)中,是Java7中提供的用于执行并行任务的框架,其可以将大任务分割成若干个小任务,最终汇总每个小任务的结果后得到最 ...

  7. Python中字符前添加r,b,u,f前缀的含义

    1.在python字符串前添加r,意思为消除转义字符 2.在python字符串前添加f,意思为支持大括号内的python 表达式. 3.在python字符串前添加b,意思为字符串类型为byte类型,在 ...

  8. JVM学习-自动内存管理

    文章原文:https://gaoyubo.cn/blogs/6997cf1f.html 一.运行时数据区 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域.这些区域 ...

  9. MB01 BAPI_GOODSMVT_CREATE退货

    "-----------------------------------------@斌将军--------------------------------------------DATA: ...

  10. Vue | vuex安装失败解决的方法实例

    Vuex是一个专为Vue.js应用程序开发的状态管理模式 下面这篇文章主要给大家介绍了关于vuex安装失败解决的方法,文中通过图文介绍的非常详细,需要的朋友可以参考下 1.报错信息: npm ERR! ...