makefile脚本基本语法

2023-11-18

l Makefile脚本流程

Shell脚本——make命令和Makefile

make命令是一个常用的编译命令,尤其在C/C++开发中,make命令通过makefile文件中描述源程序之间的依赖关系进行自动编译;makefile文件是按照规定格式编写,需说明如何编译各个源文件并连接生成可执行文件,并要求定义源文件之间的依赖关系;首次执行make命令时,编译所有相关文件,之后再执行make命令时,以增量方式进行编译,即只对修改的源文件相关的目标文件进行编译;

注:许多tarball格式的开源软件,解压后先执行./configure,再执行make,然后执行make install进行安装;

makefile文件支持include,即把一些基本依赖规则写在一个公共文件中,其他makefile文件包含此文件;

通常公共makefile文件命名为common.mk;

一、make命令

make命令后接参数,称为目标;

1 常见目标

make all:编译所有目标

make install:安装已编译的程序

make uninstall:卸载已安装的程序

make clean:删除由make命令产生的文件,通常删除目标文件.o

make distclean:删除由./configure产生的文件

make check:测试刚编译的软件

make installcheck:检查安装的库和程序

make dist:重新打包成packname-version.tar.gz

执行make命令时,需要一个Makefile文件,以告诉make命令如何编译和链接程序;

2 参数

-B:重新建立所有目标

-d:打印调试信息

-C:切换到指定路径下寻找Makefile

-f:将指定文件看做Makefile

-j:同时运行命令的个数,即多线程执行Makefile,后接的个数可由nproc命令返回值来指定

注:nproc命令打印当前进程可用的处理数(线程数);

二、程序的编译和链接

一般在C/C++开发中,首先将源文件编译成目标文件(Windows下.obj文件,Unix下.o文件)——编译compile,再将目标文件合成执行文件——链接link;

三、make命令如何工作?

1 make在当前目录下寻找“Makefile”或“makefile”文件

2 若找到,查找文件中的第一个目标文件.o

3 若目标文件不存在,根据依赖关系查找.s文件

4 若.s文件不存在,根据依赖关系查找.i文件

5 若.i文件不存在,根据依赖关系查找.c文件,此时.c文件一定存在,于是生成一个.o文件,再去执行

四、Makefile文件格式

1 概述

Makefile文件由一系列规则rules构成,每条规则形式如下:

<target>: <prerequisites>

[Tab]<commands>

第一行冒号前为目标,冒号后为前置条件;第二行必须由一个Tab键起首,后接命令;目标是必须的,不可省略;前置条件和命令是可选的,但两者必须至少存在一个;每条规则明确两件事——构建目标的前置条件是什么?如何构建?

2 目标target

目标可以是文件名,指明make命令所要构建的对象;也可以是某个操作名称,称“伪目标”;

clean:

       rm *.o

以上代码目标是clean,命令是rm *.o;执行make clean命令,实现对象文件的删除;为避免设置的伪目标名称在当前路径下有相同名称的文件,make命令发现该名称的文件已存在,便不再构建,也就不执行rm操作的情况发生,先将该名称声明为伪目标,因此make命令不会检查是否存在该名称的文件,每次执行对应的操作;

.PHONY:clean

clean:

       rm *.o

若make命令没有指定目标,默认执行Makefile文件中第一个目标;

3 前置条件prerequisites

前置条件通常是一组文件名,用空格隔开;指定目标是否重新构建的判断标准——只要有一个前置条件不存在或有更新,则该目标需重新构建;

result.txt:source.txt

cp source.txt result.txt

若当前路径下source.txt存在,make result.txt可正常执行,否则需再写一条规则,用于生成source.txt;

source.txt:

       echo "This is a source file." > source.txt

source.txt没有前置条件,与其他文件文官,只要该文件不存在,每次执行make source.txt命令都会生成该文件;

若要生成多个文件,写法如下:

source:file1 file2 file3

source是伪目标,只有3个前置条件,没有对应命令;执行make source命令后一次性生成file1 file2 file3文件,比如下写法方便:

make file1

make file2

make file3

4 命令commands

命令表示如何更新目标文件,由一行或多行shell命令组成;

注:

shell命令一定是写在命令中,否则会被make忽略;每行命令前必须有一个Tab键;每行命令在一个独立的shell中执行,shell之间没有继承关系,因此上一行为的变量赋值,在下一行无效;若前后两条命令有共享数据,可写在一行,用分号隔开;

var-kept:

       export foo=bar;echo "foo=[$$foo]"或,在换行符前加反斜杠\进行转义;

var-kept:

       export foo=bar;\

       echo "foo=[$$foo]"或,加上.ONSHELL命令;

.ONESHELL:

var-kept:

       export foo=bar;

       echo "foo=[$$foo]"

五、Makefile文件语法

1 注释

每行以#开头的为注释;

2 回声echoing

正常情况下,make打印每条命令,再执行该命令,称回声;在命令前加@,关闭回声,即只输出命令的执行结果,出错则停止执行;

注:

前缀-表示命令执行有错,忽略错误,继续执行;不加前缀输出执行的命令和命令执行的结果,出错则停止执行;

3 通配符

通配符指定一组符合条件的文件;Makefile通配符与bash一致;

*:任意0个或多个字符

 

?:任意一个字符

[...]:方括号内任意一个字符

4 模式匹配

make命令允许对文件名进行类似正则运算的匹配,主要用到%;

5 变量和赋值符

自定义变量,使用=赋值;调用变量,将变量名放在$()中;

Makefile提供四种赋值运算符——=、:=、?=、+=;

(1)=

递归展开赋值,默认赋值方式;

var2=$(var1)

var1="TEST"

all:

       echo $(var2)

输出:TEST

(2):=

直接赋值,不会递归展开,若引用的变量不存在,则为空串;

var2:=$(var1)

var1="TEST"

all:

       echo $(var2)

输出:(空串)

(3)?=

若未初始化,则赋值;

var1="test"

var1?="TEST"

var2?="TEST"

all:

       echo $(var1) and $(var2)

输出:test and TEST

(4)+=

将值追加到现有内容末尾;

var1="TEST"

var1+="test"

all:

       echo var1

输出:TESTtest

六、内置变量

make命令提供一系列内置变量,如$(CC)指向当前使用的编译器,$(MAKE)指向当前使用的make工具;

RM:rm -f

AR:ar

CC:cc

CXX:g++

七、自动变量

自动变量的值与当前规则有关;

1 $@

表示当前目标;

2 $<

表示第一个前置条件;

3 $?

表示所有比目标更新的前置条件;

4 $^

表示所有前置条件;

5 $(@D)和$(@F)

$(@D)表示$@的目标名,$(@F)表示$@的文件名;

八、判断

Makefile文件使用bash语法,完成判断;

<条件语句>

<条件为真,执行程序段>

else

<条件为假,执行程序段>

endif

1 条件语句

(1)ifeq 比较两个参数值是否相等

ifeq (arg1, arg2)

ifeq 'arg1' 'arg2'

ifeq "arg1" "arg2"

ifeq 'arg1' "arg2"

ifeq "arg1" 'arg2'

注:参数还可用make函数,如ifeq ($(strip $(foo)),);

(2)ifneq 比较两个参数值是否不等

ifneq (arg1, arg2)

ifneq 'arg1' 'arg2'

ifneq "arg1" "arg2"

ifneq 'arg1' "arg2"

ifneq "arg1" 'arg2'

(3)ifdef 判断变量是否有值

ifdef var

(4)ifndef 判断变量是否无值

ifndef var

举例1:

bar=

foo=$(bar)

ifdef foo

frobozz=yes

else

frobozz=no

endif

#frobozz值为yes

举例2:

foo=

ifdef foo

frobozz=yes

else

frobozz=no

endif

#frobozz值为no

注:以上举例说明ifdef只是测试一个变量是否有值,而非把变量扩展到当前位置!

九、循环

Makefile文件使用bash语法,完成判断;

1 写法一

LIST变量是Makefile变量,引用Makefile变量需使用$()括起来;

而all目标后的命令是shell命令,其中定义的变量也是shell变量,引用shell变量需使用$$作为开头,但shell变量不需括号;

LIST = one two three

all:

       for i in $(LIST); do \

              echo $$i; \

       done

2 写法二

all:

       for i in one two three; do \

              echo $$i; \

       done

十、函数

 

1 函数的调用语法

函数调用,类似变量使用,语法如下:

$(func args) # 推荐!

# 或

${func args}

其中,args参数之间以","隔开,func和args之间以" "隔开;

2 字符串处理函数

(1)$(subst <from>,<to>,<text>)

字符串替换,将字符串<text>中的<from>转换成<to>,返回替换后的字符串

(2)$(patsubst <pattern>,<replacement>,<text>)

模式字符串替换,查找字符串<text>中符合模式<pattern>的单词,并替换成<replacement>;

注:<pattern>可包括通配符%,表示任意长度的字符串;

若<replacement>也包括%,则这里的%是<pattern>中%代表的字符串;

(3)$(strip <string>)

去空格,去掉<string>中开头和结尾的空白符,返回去掉空格的字符串;

(4)$(findstring <find>,<in>)

查找字符串,在字符串<in>中查找<find>字符串,若找到则返回<find>,否则返回空字符串;

(5)$(filter <pattern...>,<text>)

过滤,以<pattern>模式过滤<text>字符串中的单词,保留符合模式的单词,返回符合模式的字符串;可有多个模式,模式之间以" "隔开;

(6)$(filter-out <pattern...>,<text>)

反向过滤,以<pattern>模式过滤<text>字符串中的单词,去掉符合模式的单词,返回不符合模式的字符串;可有多个模式,模式之间以" "隔开;

(7)$(sort <list>)

排序,给字符串<list>中的单词以升序排序,返回排序后的字符串;

(8)$(word <n>,<text>)

取单词,获取并返回字符串<text>中第<n>个单词(从1开始),若<n>比<text>中的单词数大,则返回空字符串;

(9)$(wordlist <s>,<e>,<text>)

取单词串,获取并返回字符串<text>中第<s>个到第<e>个的单词串;

(10)$(words <text>)

单词个数统计,统计并返回字符串<text>中单词个数;

注:获取字符串中最后一个单词,使用$(word $(words <text>),<text>)

(11)$(firstword <text>)

首单词,获取并返回字符串<text>中第一个单词;

注:等价于$(word 1,<text>)

3 文件名处理函数

(1)$(dir <names...>)

取目录,从文件名序列<names>中获取并返回目录部分;

注:目录部分指最后一个反斜杠之前的部分,若无反斜杠,则返回"./";

(2)$(notdir <names...>)

取文件,从文件名序列<names>中获取并返回文件部分;

注:文件部分指最后一个反斜杠之后的部分;

(3)$(suffix <names...>)

取后缀,从文件名序列<names>中获取并返回各文件的后缀序列,若无后缀,则返回空字符串;

(4)$(basename <names>)

取前缀,从文件名序列<names>中获取并返回各文件的前缀序列,若无前缀,则返回空字符串;

(5)$(addsuffix <suffix>,<names...>

加后缀,把<suffix>加到<names>中每个单词后面并返回;

(6)$(addprefix <prefix>,<names...>)

加前缀,把<prefix>加到<names>中每个单词前面并返回;

(7)$(join <list1>,<list2>)

连接,将<list2>中的单词对应的加到<list1>中单词之后;

(8)$(wildcard <pattern...>)

拓展通配符,用于定义变量和引用函数时通配符失效的情况,获取符合模式的字符串并返回;

4 其他函数

(1)$(foreach <var>,<list>,<text>)

把<list>中单词逐一取出放到<var>所指定的变量中,再执行<text>所包含的表达式,每次执行<text>都会返回一个字符串,执行完foreach后返回由多个字符串组成、空格隔开的字符串;

### 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

(2)$(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

(3)$(shell <shell-command>)

执行一个shell命令,将执行结果作为函数返回;

(4)make控制函数

$(error <text>):产生一个致命错误,Makefile停止执行

$(warning <text>):输出警告信息,Makefile继续执行

(5)$(call <expression>,<para1>,<para2>,...)

<expression>表达式中的变量,如$(1)、$(2)、...等被参数<para1>、<para2>、...依次取代,<expression>的返回值就是call函数的返回值;

### Makefile 内容

log = "====debug====" $(1) "====end===="

all:

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

### bash 中执行 make

$ make

====debug==== 正在 Make ====end====

 

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

makefile脚本基本语法 的相关文章

  • 21. Merge Two Sorted Lists

    题目 Merge two sorted linked lists and return it as a new list The new list should be made by splicing together the nodes
  • TensorRT学习(二)通过C++使用

    本文源于学习TensorRT文档 TensorRT Developer Guide 第2章 WORKING WITH TENSORRT USING THE C API 的理解 一 TensorRT实例化对象 使用TensorRT进行推理需要
  • 为什么要接入高防IP,到底有什么作用呢

    首先我们要了解什么是高防IP DDOS高防IP是为了应对互联网的DDOS攻击而产生的一款付费增值产品 根据不同的业务接入的方法也不同 在开通这款DDOS高防IP时服务商会给到对应的高防IP作为对外IP和业务IP使用 游戏业务就需要将之前配置
  • rknn-toolkit安装的一点感想

    rknn toolkit装了好多次 总是会报错 终于尝试出了一个版本可以用的 注意windows和linux上都能用 rknn toolkit v1 7 1 cuda 10 1 python 3 6 pytorch 1 5 1 opencv
  • Ubuntu Hadoop分布式部署

    Ubuntu Hadoop分布式部署 1 vim usr local hadoop 3 2 2 etc hadoop hadoop env sh usr local hadoop 3 2 2为Hadoop安装目录 export JAVA H
  • Android-使用WebView视图显示网页

    简介 Android提供了WebView组件 相关类为android webkit WebView android webkit WebViewClient 表面上来看 这个组件与普通的ImageView差不多 但实际上 这个组件的功能要强
  • ProGuard参数使用说明

    ProGuard 使用说明 本文主要是翻译proguad的官方文档 以便以后使用的时候不需要再次去看英文 每次写proguard总是那么痛苦 必须写个博客记录一下 有些地方的意思我也不懂 有注明原文 什么是proguard proguard
  • 微信程序 自定义遮罩层遮不住底部tabbar解决

    一 先上效果 二 方法 1 自定义底部tabbar 实现 https developers weixin qq com miniprogram dev framework ability custom tabbar html 官网去抄 简单
  • 使用Flask开发简单接口

    使用Flask开发简单接口 作为测试人员 在工作或者学习的过程中 有时会遇到没有可以调用的现成的接口 导致我们的代码没法调试跑通的情况 这时 我们使用python中的web框架Flask就可以很方便的编写简单的接口 用于调用或调试 在之前的
  • 数据挖掘实验(四):决策树归纳 R语言

    一 实验目的 决策树分类算法 decision tree 通过树状结构对具有某特征属性的样本进行分类 其典型算法包括ID3算法 C4 5算法 C5 0算法 CART算法等 本次实验掌握用ID3的信息增益来实现决策树归纳 二 实验软件 Rst
  • java中单列集合的根接口是_java 单列集合总结

    Collection 接口 add remove contains clear size 迭代器遍历 普通迭代器 不能再遍历过程中修改集合的长度 List接口 单列集合 有序可重复 有索引 add index obj remove inde
  • keras中的sequential模型

    序贯模型 Sequential 单输入单输出 一条路通到底 层与层之间只有相邻关系 没有跨层连接 这种模型编译速度快 操作也比较简单 model Sequential Sequential模型的核心操作是添加layers 图层 以下展示如何
  • LVM条带卷

    4 4 2 创建条带卷 如果有大量连续读 写操作 创建条带逻辑卷可提高数据 I O 的效率 有关条带卷的常规信息 请查看 第 2 3 2 节 条带逻辑卷 创建条带逻辑卷时 可使用 lvcreate 命令的 i 参数指定条带数 这样就决定了逻
  • 数据库中delete和drop的区别

    delete 作用于数据上 即对数据进行删除 不修改表结构 例 delete from STU where sno S001 表示从学生表中删除学号为S001的学生 此过程并不破坏表结构 drop 可对数据库 表以及字段进行修改 操作涉及修
  • Linux·进程权限控制

    Linux系统的安全性得益于其进程权限和文件权限的控制机制 今天抽空梳理下Linux下的进程权限控制相关的文件权限涉及一点 首先明确四个名词 真实用户ID real ID 有效用户ID effective ID 保存用户ID Saved I
  • list泛型总结 Map<String, Collection<?>> result = new HashMap<>(6);集合list1 instanceof Collection

    package com gang import java util public class Harbor007 public static void main String args Calendar calendar Calendar
  • 查看python已安装模块的方法小结

    随着使用python的时间越来越长 安装的python模块也越来越多 时间久了都不记得自己之前到底对自己的电脑做过些什么了 于是乎就想要查看一下自己安装的python模块 现将查看方法总结如下 一 命令行下使用pydoc命令 在命令行下运行
  • 2023年人工智能GPT-4时代,最新13个ChatGPT商业市场AIGC应用正在掀起革命性变革!

    目录 前言 ChatGPT商业应用 LLM是星辰大海 1 研究背景 1 1 研究背景 1 2 研究方法 2 商业应用和案例分析 2 1 工具层 ChatGPT 搜索 ChatGPT 办公 ChatGPT 教育 2 2 行业层 ChatGPT
  • Pyramid Vision Transformer: A Versatile Backbone for Dense Prediction without Convolutions

    A 问题 首先 对于dense prediction tasks 完全无卷积的的transformer backbone少有人研究 而VIT作为用在图像分类任务的完全transformer结构 很难直接应用于像素级别的dense predi

随机推荐

  • JAVA中io流常见异常_javaAPI_IO流基础_异常

    异常 1 异常的概述和分类 java中的异常有一个超类Throwable 然后其有俩个子类接口Error和Exception 其中Error是严重问题 这一个是程序中无法解决的 而另一个 Exception则是一般问题 Exception又
  • Android 模拟器打开错误 :x86 emulation currently requires hardware acceleration!

    第一次配置完成 Android 下载 模拟器进行打开使用时 发生错误 x86 emulation currently requires hardware acceleration 如图所示 问题分析 x86 emulation 仿真器 模拟
  • Vue之监听属性

    一 什么是监听属性 监听属性是Vue js提供的一种用来监听和响应Vue实例中的数据变化的方式 在监听数据对象中的属性时 每当监听的属性发生变化 都会执行特定的操作 监听属性可以定义在watch选项中 也可以使用实例方法vm watch 在
  • C++变量声明定义

    1 extern 声明变量 在一个文件里声明以后 表示该变量要去其它文件找变量 告诉编译器 你现在编译的文件中 有一个标识符虽然没有在本文件或本文件当前位置中定义 但是它是在别的文件中或本文件其它位置定义的全局变量 你要放行 声明变量 ex
  • word中行距设置固定值,图片显示不全的问题

    撰写毕业大论文时 要求正文行间距设置为20磅固定值 但会出现 粘贴图片时 图片显示不全 软件也把图片做一行的固定值显示 解决办法是 选中图片 按ctrl 1单倍行距组合键 此时图片就能显示全了 图片的行间距就是单倍行距
  • 腾讯采用Intertrust的Marlin DRM方案保护其在线视频服务

    转自 http tech ifeng com internet detail 2013 01 11 21094087 0 shtml 1月10日 Intertrust公司今天宣布 已授权腾讯使用Marlin DRM方案保护腾讯在线视频平台的
  • 5.DML语句

    DML语句用于操作数据表的数据 如 插入 修改 删除 insert into update和delete from三个命令组成 1 insert into INSERT INTO grade id math VALUES 1 83 若省略表
  • Android下设置CPU核心数和频率

    现在的Android手机双核 四核变得非常普遍 同时CPU频率经常轻松上2G 功耗肯定会显著增加 而大多数的ARM架构的CPU采用的是对称多处理 SMP 的方式处理多CPU 这就意味着每个CPU核心是被平等对待的 同时打开又同时关闭 显然
  • python中的eval

    eval将字符串作为执行命令 题目 https www hackerrank com challenges py set discard remove pop 也可以使用getattr a b 相当于a b 代码如下 n int input
  • vue中如何监听localStorage中值的变化

    一 说明 在日常开发中 我们经常使用localStorage来存储一些变量 这些变量会存储在浏览中 对于localStorage来说 即使关闭浏览器 这些变量依然存储着 方便我们开发的时候在别的地方使用 二 localStorage的使用
  • Java集合的使用

    集合的使用 集合框架的概述 数组在存储多个数据的特点 数组在存储多个数据的缺点 Java 集合可分为 Collection 和 Map 两种体系 Collection接口 Map接口 Collection 接口的使用 说明 常用方法 案例一
  • java hashcode() 和equals()详解 以及set不能重复问题

    1 首先equals 和hashcode 这两个方法都是从object类中继承过来的 equals 方法在object类中定义如下 public boolean equals Object obj return this obj 很明显是对
  • Qt——用于表格QTableView的模型

    如果想使用表格来呈现数据 Qt提供了一个方便的部件QTableWidget 但是直接用它实现一些功能可能比较困难 这里将介绍一种强大 灵活的方式来操作表格 一 模型 视图架构 在这个架构中 模型用于存储数据 视图用于呈现数据 除此之外 还有
  • Windows Linux 子系统安装步骤

    准备 一 启用 适用于Linux的Windows子系统 功能 方法一 Win10搜索 控制面板 打开 选择 程序和功能 项 左上角 启用或关闭Windows功能 翻到最下方 选择 适用于Linux的Windows子系统 确定后按提示操作即可
  • VC++ CComboBox自绘阶段性总结

    在绘制列表框左侧的时候 本来说是采用的CImageList接口进行绘制 结果发现绘制后的图标不干净 有锯齿或者黑色背景存在 于是采用了GDI 来实现左侧图标的绘制 代码如下 GDI是完全可以通过图像处理彻底解决图标不干净问题的 之前文章有介
  • LeetCode 88 合并两个有序数组

    LeetCode 88 合并两个有序数组 给你两个按 非递减顺序 排列的整数数组 nums1 和 nums2 另有两个整数 m 和 n 分别表示 nums1 和 nums2 中的元素数目 请你 合并 nums2 到 nums1 中 使合并后
  • 0基础学python——文件操作

    这一小节的内容主要是python的文件操作函数和相关内容 文件 就是把一些数据存放起来 可以让程序下一次执行的时候直接使用而不必重新制作一份 gt 文件类型 文本文件 以文字存储为主 读写均以 字符 为单位 二进制文件 以图形 声音 影像为
  • 解决python urllib3报错urllib3.exceptions.MaxRetryError xxx [SSL: CERTIFICATE_VERIFY_FAILED] xxx

    完整报错如下 urllib3 exceptions MaxRetryError HTTPSConnectionPool host xxxxx port 443 Max retries exceeded with url xxxxxxx Ca
  • platform总线、设备、驱动模型之led驱动实例

    在 Linux 2 6 的设备驱动模型中 关心总线 设备和驱动这 3 个实体 总线将设备和驱动绑定 在系统每注册一个设备的时候 会寻找与之匹配的驱动 相反的 在系统每注册一个驱动的时候 会寻找与之匹配的设备 而匹配由总线完成 注意 所谓的p
  • makefile脚本基本语法

    l Makefile脚本流程 Shell脚本 make命令和Makefile make命令是一个常用的编译命令 尤其在C C 开发中 make命令通过makefile文件中描述源程序之间的依赖关系进行自动编译 makefile文件是按照规定