一、温故知新

1、可执行程序的生成过程

2、gcc的常用操作

gcc/g++ 命令的基本格式为
gcc -[选项] [文件名]
例如:
gcc -c -I /home/inc/ test.c -o test.o
gcc -I /home/inc/ -O2 -g3 main.c test.o -o main

二、make操作

make 的基本格式为:
make -[选项] [文件名]
例如:
make -v
make -n
make -C /home/erc/ -f Makefile02 -s

三、编写Makefile文件时常用操作

下面会着重介绍编写Makefile文件时经常用到的操作,对于makef的执行规则等更细致全面的知识可以参考Makefile教程(绝对经典,所有问题看这一篇足够了)

注意:在Makefile文件中 空格和缩进是完全不同的,不可以相互转换。

1、框架格式

目标1:依赖
命令
目标2:依赖
命令
目标3:依赖
命令

2、举例

假如说我现在有这么一个程序需要编译:

所需的 .h 头文件在 /home/inc/ 目录下
所需的 .c 源文件在 /home/src/ 目录下
#include "myadd.h"
#include "mysub.h" 主函数在当前目录下的 main.c 里

int main()

{

return 0;

}

那么Makefile文件应该这么编写

main:/home/inc/myadd.o /home/inc/mysub.o ./main.o
gcc -I /home/inc/ /home/inc/myadd.o /home/inc/mysub.o ./main.o -o main /home/inc/myadd.o:/home/src/myadd.c

gcc -c -I /home/inc/ /home/src/myadd.c -o /home/src/myadd.o /home/inc/mysub.o:/home/src/mysub.c

gcc -c -I /home/inc/ /home/src/mysub.c -o /home/src/mysub.o ./main.o:./main.c

gcc -c -I /home/inc/ ./main.c -o ./main.o clean:

rm -rf /home/inc/mysub.o /home/inc/myadd.o ./main.o main

3、优化

以下是优化后的代码

NAME=main

SRC=\((</span>wildcard ./*.c<span class="token variable">)</span></span> <span class="token variable"><span class="token variable">\)(wildcard /home/src/*.c)

OBJ=\((</span>patsubst %.c, %.o, <span class="token punctuation">\)(SRC))

INC=-I"/home/inc/"

\((</span>NAME<span class="token variable">)</span></span><span class="token builtin class-name">:</span><span class="token variable"><span class="token variable">\)(OBJ)

@ \((</span>CC<span class="token variable">)</span></span> <span class="token parameter variable">-g</span> <span class="token variable"><span class="token variable">\)(INC) \((</span>OBJ<span class="token variable">)</span></span> <span class="token parameter variable">-o</span> <span class="token variable"><span class="token variable">\)(NAME) %.o:%.c

@ \((</span>CC<span class="token variable">)</span></span> <span class="token parameter variable">-c</span> <span class="token variable"><span class="token variable">\)(INC) \(^ <span class="token parameter variable">-o</span> <span class="token variable">\)@ .PHONY:clean clean:

@ rm -f \((</span>OBJ<span class="token variable">)</span></span> <span class="token variable"><span class="token variable">\)(NAME)

1). 伪目标 .PHONY

声明目标成伪目标之后,make会无条件执行该目标,
且不会判断目标是否存在或者是否需要更新。

例如:常常把 clean声明为伪目标

	.PHONY:clean
clean:
<span class="token function">rm</span> <span class="token parameter variable">-rf</span> <span class="token variable"><span class="token variable">$(</span>OBJ<span class="token variable">)</span></span> <span class="token variable"><span class="token variable">$(</span>NAME<span class="token variable">)</span></span>

2). $ 和 @ 符号的作用

@ :运行命令时,隐藏命令
例如:@ rm -f $(OBJ) $(NAME) 会删除这些文件,但终端上不会有rm -r 输出

$ :使用变量,(定义变量的时候不需要加$)
例如 echo $(NAME)

3). 变量

注意:

不管是自定义变量或者是系统变量,使用方法都一样

使用方法即: $(变量名称)

变量的工作原理和 C语言中的 define类似,是会直接替换的,
所以一定要处理好空格。

变量种类

自定义变量:
定义: NAME=main
使用: echo $(NAME)

系统常量:
优点:与设备无关,全平台通用,使 makefile 可以跨平台。
常见系统常量:

Makefile 中常见系统变量:
(加 星号 的是使用频率较高的)

4). 匹配模式

目标前缀名和依赖前缀名相同时,可以使用匹配模式缩减代码长度
例如:
myadd.o : myadd.c 可以被替换为
%.o : %.c

$(wildcard /home/src/*.c)
获取某目录下的所有.c文件名称

$(patsubst %.c, %.o, $(SRC))
获取某些 .c 文件对应的 .o 文件

使用方法

获取当前目录下的所有 .c 文件的全名,获取 /home/src/目录下的所有 .c 文件的全名
并把文件名储存到 SRC 变量里
SRC=$(wildcard ./*.c) $(wildcard /home/src/*.c) 获取SRC变量里的 .c 文件对应的 .o 文件的全名

并把文件名储存到 OBJ 变量里

OBJ=\((</span>patsubst %.c, %.o, <span class="token punctuation">\)(SRC))

此时再回头看优化代码,OvO,原来是这样。

4、Makefile中的条件判断 ifeq ifdef

使用:

ifeq 判断两个变量是否相等
括号里是要判断的两个变量

A=aa
TMP:=
ifeq ($(A),aa)
TMP1:=aa
else
TMP1:=no-aa
endif

ifdef 判断某变量是否定义且赋值(如果没有赋值也是false)

B=
TMP2:=
ifdef (B)
TMP2:=def-B
else
TMP2:=ndef-B
endif

注意事项:

注意书写格式,ifeq 或者 ifdef 和括号之间有一个空格!!!

Makefile中不存在 elif 但可以嵌套调用 ifeq 或 ifdef 来实现elif的功能

5、Makefile中的循环 foreach

语法

命令 $(foreach v, 集合, 对v进行重命名或拼接等)

例子

TARGET=t1 t2 t3 t4

all:

mkdir \((</span>foreach v, <span class="token punctuation">\)(TARGET), $v_dir)

说明:

遍历TARGET变量,然后在每个变量名后面拼接_dir组成新的名称,

然后以新的名称创建目录

演示结果:

6、Makefile中的赋值 = 和 :=

= 是最终赋值,
:= 是临时赋值,
实际上 := 类似 C++中的变量引用,
对于 = 和 := 具体的区别,大家可以自己动手实操感受下。

总结

纸上谈来终觉浅,绝知此事要躬行!

[转帖]gcc与makefile常用操作(绝对常用,也绝对够用)的更多相关文章

  1. centos LAMP第四部分mysql操作 忘记root密码 skip-innodb 配置慢查询日志 mysql常用操作 mysql常用操作 mysql备份与恢复 第二十二节课

    centos  LAMP第四部分mysql操作  忘记root密码  skip-innodb 配置慢查询日志 mysql常用操作  mysql常用操作 mysql备份与恢复   第二十二节课 mysq ...

  2. JavaScript常用操作,常用类

    算术运算符 重点关注 算数,赋值,逻辑运算符,三目运算符 <!DOCTYPE html> <html> <head> <meta charset=" ...

  3. Vim常用操作(1)-常用指令

    1.清空文件内容 Normal模式下,先输入"gg",将光标定位到文件首,然后输入"dG",文件就被清空了.当然也可以直接输入"ggdG".

  4. 大白话说GIT常用操作,常用指令git操作大全

    列一下在开发中用的比较多的git指令 git clone https://github.com/chineseLiao/Small-career // 克隆远程仓库到本地 git add . // 把 ...

  5. FastJson的常用操作

    FastJson的常用操作 2017-06-05 常用操作包括以下内容: 对象与(JsonObject或JsonArray)与String的互换 String转换为(JsonObject或JsonAr ...

  6. Hbase到Solr同步常用操作

    Hbase到Solr同步常用操作 1. 整体流程 2. 常用操作 Hbase常用操作 Solr常用操作 hbase-index常用操作 3. 其他资料 Lily HBase Indexer使用整理 h ...

  7. Linux - nginx基础及常用操作

    目录 Linux - nginx基础及常用操作 Tengine淘宝nginx安装流程 nginx的主配置文件nginx.conf 基于域名的多虚拟主机实战 nginx的访问日志功能 网站的404页面优 ...

  8. gcc与makefile编译 BY 四喜三顺

    gcc编译控制过程:(假设源代码为a.c)(1)源文件到预处理文件:    gcc -E -o a.cxx a.c    a.cxx显示调用哪些头文件(2)生成汇编代码:              g ...

  9. Linux常用操作练习

    Linux常用操作练习 练习一:安装CentOS 1.设置为1G内存(才有图形界面).10G硬盘 2.分给交换分区2G(4G一下2G,8G-32G分4G-8G) 练习二:安装CentOS迷你版 1.安 ...

  10. GreenPlum-数据存储目录迁移及常用操作

    一.环境介绍 Greenplum5 3节点集群,Centos7.2虚拟机, 二.需求 因为/home目录磁盘空间已满,需要将Greenplum的数据存储目录转移到新的分区/opt目录下,虚拟机磁盘管理 ...

随机推荐

  1. 心理健康数据集:mental_health_chatbot_dataset

    一.数据集描述 1.数据集摘要   该数据集包含与心理健康相关的问题和答案的对话对,以单一文本形式呈现.数据集是从流行的医疗博客(如WebMD.Mayo Clinic和HealthLine).在线常见 ...

  2. 当线下门店遇上AI:华为云ModelBox携手佳华科技客流分析实践

    摘要:在赋能传统门店客流经营数字化转型方面,华为云ModelBox与伙伴佳华科技合作推出的"华为云客流统计项目",算是一次成功的探索. 本文分享自华为云社区<当线下门店遇上A ...

  3. 五一高铁票难抢?用RPA机器人试试!

    随着信息数字化的高速发展,RPA在各行业中得到广泛应用,热度大增.这匹"技术黑马"已然成为构建业务流程自动化的重要引擎之一,助力企业组织向"智能自动化转型. 什么是RPA ...

  4. Scala学习系列(二)——环境安装配置

    Scala下载地址:https://www.scala-lang.org/download/ 一.安装JDK 首先,因为Scala是运行在JVM平台上的,所以安装Scala之前要安装JDK 二.二进制 ...

  5. 第04讲:Flink 常用的 DataSet 和 DataStream API

    Flink系列文章 第01讲:Flink 的应用场景和架构模型 第02讲:Flink 入门程序 WordCount 和 SQL 实现 第03讲:Flink 的编程模型与其他框架比较 第04讲:Flin ...

  6. Mysql--binlog日志

    一.简介 binlog日志也称二进制日志,记录了所有的DDL和DML( 除了数据查询语句 )语句,以事件形式记录,还包含语句所执行的消耗的时间,MySQL的二进制日志是事务安全型的. 一般来说开启二进 ...

  7. L1-048 矩阵A乘以B (15分)

    给定两个矩阵A和B,要求你计算它们的乘积矩阵 \(AB\).需要注意的是,只有规模匹配的矩阵才可以相乘.即若A有 \(R_a\) 行.\(C_a\) 列,B有 \(R_b\) 行.\(C_b\) 列, ...

  8. 八、java操作swift对象存储(静态大对象)

    系列导航 一.swift对象存储环境搭建 二.swift添加存储策略 三.swift大对象--动态大对象 四.swift大对象--静态态大对象 五.java操作swift对象存储(官网样例) 六.ja ...

  9. webgl创建一个点

  10. IDEA用上这十大插件绝对舒服

    本文翻译自国外论坛 medium,原文地址:https://medium.com/@xjpp22/top-10-plugins-for-intellij-idea-you-dont-want-to-m ...