013.Makefile
目录
- Makefiel 编写
- 程序的编译和链接
- Make 工作原理
- 一个项目的规则:
- 测试代码:
- main:
- sayHello.h
- sayHello.cpp
- 问题引出
- 解决方案
- Make工作流程
- Makefile范例:
- Make变量
- 更省事的方式,让Make 自动推导
- 总结:
- Makefile思维导图:
- Makefile工作原理图
Makefiel 编写
我们之前其实已经写过一些makefile了,只是没有具体介绍,本篇博客就详细的介绍一下Makefile。
程序的编译和链接
使用C、C++编写可执行程序,首先要把源文件编译成中间代码文件,Linux下是 .o 文件,就是 Object File,这个动作叫做编译。然后再把大量的 .o文件合成执行文件,这个动作叫作链接。
一个项目,拥有成百上千的源程序文件,编译链接这些源文件都是由规则的,Makefile闪亮登场!Makefile确定整个工程的编译规则,只需要一个make命令,就可以实现“自动化编译”。make是一个解释makefile中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如:Delphi的make,Visual C++的nmake,Linux下GNU的make。
Make 工作原理
一个项目的规则:
1)如果这个工程没有编译过,那么我们的所有C文件都要编译并被链接。
2)如果这个工程的某几个C文件被修改,那么我们只编译被修改的C文件,并链接目标程序。
3)如果这个工程的头文件被改变了,那么我们需要编译引用了这几个头文件的C文件,并链接目标程序。
只要我们的Makefile写得够好,我们只用一个make命令就可以完成,make命令会自动智能地根据当前的文件修改的情况来确定哪些文件需要重编译,从而自己编译所需要的文件和链接目标程序。
Makefile的规则:
target … : prerequisites …
command
…
…
target也就是一个目标文件,可以是Object File,也可以是执行文件。还可以是一个标签(Label),对于标签这种特性,暂不叙述。
prerequisites就是,要生成那个target所需要的文件或是目标。
command也就是make需要执行的命令。(任意的Shell命令)
这是一个文件的依赖关系,也就是说,target这一个或多个的目标文件依赖于prerequisites中的文件,其生成规则定义在command中。说白一点就是说,prerequisites中如果有一个以上的文件比target文件要新的话,command所定义的命令就会被执行。这就是Makefile的规则。也就是Makefile中最核心的内容。
测试代码:
main:
#include"sayHello.h"int main(void)
{helloWorld();
}
sayHello.h
#pragma once
#include<iostream>using namespace std;void helloWorld();
sayHello.cpp
#include"sayHello.h"void helloWorld()
{cout << "Hello World!" << endl;cout << "Hello World!!" << endl;cout << "Hello World!!!" << endl;
}
问题引出
如果我们还是和在windows下一样直接执行是会出问题的,具体演示我已经在第二十篇博客做了,所以不再详述。
解决方案
为了解决哪些问题,我们就需要用到Makefile了。
Make工作流程
Makefile范例:
hello_demo : main.o sayHello.o g++ -o hello_demo main.o sayHello.o sayHello.o : sayHello.cpp sayHello.h g++ -c sayHello.cpp
main.o : main.cpp sayHello.h g++ -c main.cpp
clean : rm hello_demo main.o sayHello.o
1、make会在当前目录下找名字叫“Makefile”或“makefile”的文件。(只能是这两个名字中的一个)
2、如果找到,它会找文件中的第一个目标文件(target),在上面的例子中,他会找到“hello_demo”这个文件,并把这个文件作为最终的目标文件。
3、如果hello_demo文件不存在,或是hello_demo所依赖的后面的 .o 文件的文件修改时间要比hello_demo这个文件新,那么,他就会执行后面所定义的命令来生成hello_demo这个文件。
4、如果hello_demo所依赖的.o文件也不存在,那么make会在当前文件中找目标为.o文件的依赖性,如果找到则再根据那一个规则生成.o文件。
5、当然,我们的C文件和H文件都存在,于是make会生成 .o 文件,然后再用 .o 文件生命make的终极任务,也就是执行文件hello_demo了。
Make变量
一个Makefile里我们发现经常会由重复的内容,如上面范例中的前两行中的
main.o sayHello.o
如果我们的工程需要加入一个新的[.o]文件,那么我们好几个地方都需要修改原来的makefile。当然,我们的makefile并不复杂,所以在两个地方加也不累,但如果makefile变得复杂,那么我们就有可能会忘掉一个需要加入的地方,而导致编译失败。所以,为了makefile的易维护,在makefile中我们可以使用变量。makefile的变量也就是一个字符串,完全可以理解成C语言中的宏。
变量定义: 变量名 = 值 ##使用shell script 的语法
如:
objects = main.o sayHello.o
我们上面的makefile 就可以变成:
objects = main.o sayHello.o
hello_demo : $(objects) g++ -o hello_demo $(objects) sayHello.o : sayHello.cpp sayHello.h g++ -c sayHello.cpp
main.o : main.cpp sayHello.h g++ -c main.cpp
clean : rm hello_demo $(objects)
更省事的方式,让Make 自动推导
make很强大,它可以自动推导文件以及文件依赖关系后面的命令,于是我们就没必要去在每一个[.o]文件后都写上类似的命令,因为,我们的make会自动识别,并自己推导命令。
只要make看到一个[.o]文件,它就会自动的把[.c]文件加在依赖关系中,如果make找到一个main.o,那么hello.c,就会是hello.o的依赖文件。并且 g++ -c main.cpp也会被推导出来,于是,我们的makefile再也不用写得这么复杂。我们的新的makefile又出炉了。
objects = main.o sayHello.o
hello_demo : $(objects) g++ -o hello_demo $(objects) $(objects):sayHello.h
#sayHello.o : sayHello.cpp sayHello.h
# g++ -c sayHello.cpp
#main.o : main.cpp sayHello.h
# g++ -c main.cpp .PHONY:clean
clean : rm hello_demo $(objects)
总结:
大致学习到博主介绍的部分也就足够了,就算在做大项目中,Makefile也不需要我们写。。。
不过想要更深一步了解Makefile的可以看博主下面的内容:
Makefile思维导图:
Makefile工作原理图
013.Makefile相关推荐
- perl linux 独立运行,Perl脚本打包为独立执行程序
Linux平台打包Perl脚本为独立运行程序 一.安装打包工具 安装注意事项: 1)尽量选择与这个Perl版本兼容的相关Perl模块: 2)由于PAR-Packer-1.013对perl版本需求决定, ...
- linux内核模块Makefile的解析
Linux内核是一种单体内核,但是通过动态加载模块的方式,使它的开发非常灵活 方便.那么,它是如何编译内核的呢?我们可以通过分析它的Makefile入手.以下是 一个简单的hello内核模块的Make ...
- 浅显易懂 Makefile 入门 (12)— Makefile 常见的错误信息
1. 常见的错误信息 make 执行过程中所产生错误并不都是致命的,特别是在命令行之前存在 -.或者 make 使用 -k 选项执行时. make 执行过程的致命错误都带有前缀字符串 ***.错误信息 ...
- 浅显易懂 Makefile 入门 (10)— 嵌套执行 make、export 的使用
1. 嵌套执行 make 在一个大的工程文件中,不同的文件按照功能被划分到不同的模块中,每个模块可能都会有自己的编译顺序和规则,如果在一个 Makefile 文件中描述所有模块的编译规则,就会很乱,执 ...
- 浅显易懂 Makefile 入门 (09)— include 文件包含、MAKECMDGOALS
1. include文件包含 当 make 读取到 include 关键字的时候,会暂停读取当前的 Makefile,而是去读 include 包含的文件,读取结束后再继读取当前的 Makefile ...
- 浅显易懂 Makefile 入门 (08)— 默认 shell (/bin/sh)、命令回显、make参数(-n 只显示命令但不执行,-s 禁止所有回显)、单行命令、多行命令、并发执行
1. shell 相关 1.1 默认 shell Makefile 所使用的命令是由 shell 命令行组成,他们是一条一条执行的. 多个命令之间要使用分号隔开,Makefile 中的任何命令都要以 ...
- 浅显易懂 Makefile 入门 (07)— 其它函数(foreach 、if、call、origin )
1. foreach 函数 foreach 函数定义如下: $(foreach <var>,<list>,<text>) 函数的功能是:把参数 <list&g ...
- 浅显易懂 Makefile 入门 (06)— 文件名操作函数(dir、notdir、suffix、basename、addsuffix、addperfix、join、wildcard)
编写 Makefile 的时候,很多情况下需要对文件名进行操作.例如获取文件的路径,去除文件的路径,取出文件前缀或后缀等等. 注意:下面的每个函数的参数字符串都会被当作或是一个系列的文件名来看待. 1 ...
- 浅显易懂 Makefile 入门 (03)— 目标文件搜索(VPATH 和 vpath 的区别和使用)、隐含规则
1. 目标文件搜索(VPATH和vpath) 如果需要的文件是存在于不同的路径下(即源文件与 Makefile 文件不在同一个路径下),在编译的时候就用到了 Makefile 中为我们提供的目录搜索文 ...
最新文章
- 分布式存储 Ceph 的演进经验 · SOSP 2019
- 简单易懂,ThreadPoolExecutor参数说明
- MyBatis中多表查询(业务代码方式)
- 06-图1 列出连通集 (25 分)
- linux7配置网卡绑定,CentOS7双网卡绑定配置
- 2012年7月份第4周51Aspx源码发布详情
- 06_Jedis完成MySQL的条件查询案例
- 15款免费的 HTML5/CSS3 响应式网页模板
- 大括弧之战 代码风格
- 大学英语四六级写作模板
- Visual Studio报错:由于代码已经过优化或者本机框架位于调用堆栈之上,无法计算表达式的值...
- windows PAE扩展和AWE编程
- 什么是GNSS测试?如何进行GNSS测试?
- 爬虫案例——模拟登录QQ空间
- STM32HAL库微秒延时函数的实现---DWT和SysTick
- 同为iOS开发者为什么有人4k有人40k?原因就在这里!
- 石墨笔记, Onenote 和 Effie 哪个适合单口或脱口秀作者?
- ArcGIS10.8下载及安装教程(附安装步骤)
- Android AVD Manager - 安卓虚拟机的设置
- in addition to 和 except for
热门文章
- 打开旧Flutter项目说:Your Flutter application is created using an older version of the Android embedding
- 设置git客户端不经过代理
- Python菜鸟入门:day09数据结构
- 倒计时2天!1024 程序员节全日程曝光,105 场深度演讲点燃数字经济新时代
- 邹欣对话图灵奖得主Jeffrey Ullman:数据库不会进入周期性的坏循环|《新程序员》...
- 连续被巨头青睐,SpaceX 部分新项目原型系 Rust 构建
- 24K 内存上诞生的操作系统,是如何改变计算机世界的?
- 如何获取高质量 CV 训练数据?这个超详细上手教程不容错过
- 当你使用微信和QQ时,请不要忘记ICQ这个伟大的公司!
- 萌妹子语音在线吹彩虹屁陪你写代码!一个神奇的 VSCode 插件