makefile入门第一课

百度百科makefile词条
一个工程中的源文件不计其数,其按类型、功能、模块分别放在若干个目录中。
makefile 定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,
甚至于进行更复杂的功能操作,因为 makefile 就像一个 Shell 脚本一样,其中也可以执行操作系统的命令。

简单了解 g++

g++ 是 GNU 的 C++ 编译器

用 g++ 编译单个文件生成可执行文件

在 vim 中编写下列代码,保存在 main.cpp 文件中:

// in main.cpp
#include <iostream>
using namespace std; int main()
{
cout << "hello, g++!" << endl;
return 0;
}

在该文件夹下,执行下列语句编译该文件为执行文件:

g++ main.cpp -o hello

这句命令是编译 main.cpp 文件, 并将其输出文件命名为 hello
此时,可以发现,当前文件夹生成了可执行文件 hello,输入下列语句执行该文件:

./hello

此时,shell 输出 hello, g++!

用 g++ 通过编译中间文件生成可执行文件

可执行文件的生成一般包括编译链接,首先编译生成中间目标文件,然后将各个目标文件链接成可执行文件:

用 vim 编写下列代码,分别保存在不同的文件下:

  • hello.h
 #ifndef SRC_HELLO_H
#define SRC_HELLO_H
class hello
{
public:
hello();
~hello();
};
#endif // SRC_HELLO_H
  • hello.cpp
#include "hello.h"
#include <iostream> using std::cout;
using std::endl; hello::hello()
{
cout << "hello, object!" << endl;
} hello::~hello()
{
cout << "goodbye, obejct!" << endl;
}
  • main.cpp
#include <iostream>
#include "hello.h"
using namespace std; int main()
{
hello h;
return 0;
}
  1. 在当前路径输入下列语句,编译 hello.cpp,生成中间目标文件 hello.o

    g++ -c hello.cpp

    此时,当前目录生成 hello.o 文件。

  2. 在当前路径输入下列语句,编译 main.cpp,生成中间目标文件 main.o

    g++ -c main.cpp

    此时,当前目录生成 main.o 文件。

  3. 在当前路径输入下列语句,链接 hello.omain.o,生成可执行文件 hello

    g++ main.o hello.o -o hello

    此时,当前文件夹生成可执行文件 hello

  4. 在当前路径输入下列语句,执行 hello 程序:

    ./hello

    此时控制台输出为:

    hello, object!
    goodbye, obejct!

makefile 编译文件

makefile 的基本结构

makefile 由下面的基本结构组成:

target …  :  prerequisites …
recipe

taget 通常是待生成的可执行文件目标文件,也可以是标签,
prerequisites 是用来生成待生成文件的文件,即待生成文件所依赖的文件,
recipemake 命令需要执行的动作(action), 通常是不止一条的 shell 命令。

makefile 编译单个文件生成可执行文件

在当前路径下,创建下面的 .cpp 文件:

// in main.cpp
#include <iostream>
using namespace std; int main()
{
cout << "hello, makefile!" << endl;
return 0;
}

用 vim 创建文件名为 makefile 的文件:

# in makefile
#可执行文件 hello 由 main.cpp 编译得到
hello: main.cpp
# 打印一句话
echo "Begin to compile..."
# 编译 main.cpp 生成可执行文件 hello 的命令行
g++ main.cpp -o hello
# 打印一句话
echo "Has been Compiled..."

在当前路径下,输入并执行 make 命令,控制台输出:

Begin to compile...
Has been compiled...

此时,当前路径下生成可执行文件 hello ,执行 hello 文件:

./hello

此时,控制台输出:

hello, makefile!

makefile 编译中间文件并链接为可执行文件

在当前路径下,创建下面的 .cpp 文件和 .h 文件:

  • hello.h
#ifndef SRC_HELLO_H
#define SRC_HELLO_H class hello
{
public:
hello();
~hello();
}; #endif // SRC_HELLO_H
  • hello.cpp
#include "hello.h"
#include <iostream> using std::cout;
using std::endl; hello::hello()
{
cout << "hello, makefile!" << endl;
} hello::~hello()
{
cout << "goodbye, makefile!" << endl;
}
  • main.cpp
#include <iostream>
#include "hello.h"
using namespace std; int main()
{
hello h;
return 0;
}

用 vim 创建文件名为 makefile 的文件:

#可执行文件 hello 由中间目标文件 main.o 和 hello.o 链接得到
hello: main.o hello.o
# 打印一句话
echo "Begin to link main.o and hello.o..."
# 链接 main.o 和 hello.o 生成可执行文件 hello 的命令行
g++ main.o hello.o -o hello
# 打印一句话
echo "main.o and hello.o have been linked..."
#中间目标文件 hello.o 由 hello.cpp 编译得到
hello.o: hello.cpp
# 打印一句话
echo "Begin to compile hello.o..."
# 编译 hello.cpp 生成中间目标文件 hello.o
g++ -c hello.cpp
# 打印一句话
echo "hello.o has been compiled..."
main.o: main.cpp
# 打印一句话
echo "Begin to compile main.o..."
# 编译 main.cpp 生成中间目标文件 main.o
g++ -c main.cpp
# 打印一句话
echo "main.o has been compiled..."

在当前路径下,输入并执行 make 命令,控制台输出:

Begin to compile main.o...
main.o has been compiled...
Begin to compile hello.o...
hello.o has been compiled...
Begin to link main.o and hello.o...
main.o and hello.o have been linked...

此时,当前路径下生成中间目标文件 hello.omain.o 和可执行文件 hello ,执行 hello 文件:

./hello

此时,控制台输出:

hello, makefile!
goodbye, makefile!

makefile 删除编译过程中产生的中间目标文件

在上一 part 的 makefile 文件末尾加上 clean 标签及相关清理中间目标文件的,命令行,就可以在生成最终文件后自动删除相关中间文件:

clean:
rm -f main.o hello.o

则原文件变为:

#可执行文件 hello 由中间目标文件 main.o 和 hello.o 链接得到
hello: main.o hello.o
# 打印一句话
echo "Begin to link main.o and hello.o..."
# 链接 main.o 和 hello.o 生成可执行文件 hello 的命令行
g++ main.o hello.o -o hello
# 打印一句话
echo "main.o and hello.o has been linked..."
#中间目标文件 hello.o 由 hello.cpp 编译得到
hello.o: hello.cpp
# 打印一句话
echo "Begin to compile hello.o..."
# 编译 hello.cpp 生成中间目标文件 hello.o
g++ -c hello.cpp
# 打印一句话
echo "hello.o has been compiled..."
main.o: main.cpp
# 打印一句话
echo "Begin to compile main.o..."
# 编译 main.cpp 生成中间目标文件 main.o
g++ -c main.cpp
# 打印一句话
echo "main.o has been compiled..."
# 清理中间目标文件
clean:
rm -f main.o hello.o

执行 make 命令后,显式执行 make clean。此时当前路径下只有可执行文件 hello,而没有相关中间目标文件。

后记

本文主要是为从零开始学习 makefile 抛砖引玉, 相关内容尽量简单处理,详细介绍,让新手可以跟着一步一步的操作。
因此,本文可能对有一定基础的童鞋来说有一点冗长。见谅。

另外,点击下列参考文献的名称就可以跳转到相关下载页面。相关细节,参考文献中有具体的介绍。

参考文献

makefile 入门第一课的更多相关文章

  1. Asp.Net Web API 2(入门)第一课

    Asp.Net Web API 2(入门)第一课   前言 Http不仅仅服务于Web Pages.它也是一个创建展示服务和数据的API的强大平台.Http是简单的.灵活的.无处不在的.你能想象到几乎 ...

  2. emacs 入门第一课:Emacs里的基本概念

    Table of Contents 无聊的开场白 buffer(缓冲区) window(窗口)与frame Emacs的mode Emacs Lisp 函数function.命令command.键绑定 ...

  3. Android入门第一课之Java基础

    通知:由于本周六场地申请没通过,所以本周的培训临时取消. 今天给大家带来的是Android入门的第一课,由于教室申请的不确定性,因此,每次培训的内容都会在博客先提前释放出来.首先Android的APP ...

  4. Docker入门 第一课 --.Net Core 使用Docker全程记录

    微服务架构无疑是当前最火热的开发架构,而Docker作为微服务架构的首选工具,是我们必须要了解掌握的. 我通过一天的时间,网上查文档,了解基础概念,安装Docker,试验Docker命令,通过Dock ...

  5. Kotlin入门第一课:从对比Java开始

    1. 介绍 今年初,甲骨文再次对谷歌所谓的安卓侵权使用Java提起诉讼,要求后者赔偿高达90亿美元.随后便传出谷歌因此计划将主力语言切换到苹果主导的Swift,不过这事后来没了跟进. 但谷歌在这两天的 ...

  6. Spring入门第一课:Spring基础与配置Bean

    1.入门 Spring是简化java开发的一个框架,其中IoC和AOP是Spring的两个重要核心.由于Spring是非侵入性的,通过Ioc容器来管理bean的生命周期,还整合了许多其他的优秀框架,所 ...

  7. JavaScrip 入门第一课

    一.代码引入的三种方式 1.直接在head中书写 在head标签里面可以写,在body标签里面也可以写,放到head标签里面和放到body标签里面到底有什么区别,我们后续在讲~ <head> ...

  8. python 语言学入门第一课必看:编码规范

    命名 module_name, package_name, ClassName, method_name, ExceptionName, function_name, GLOBAL_VAR_NAME, ...

  9. 1、C#入门第一课

    C# 读作C Sharp,所以程序文件的扩展名为.cs 新建项目-窗体应用程序 所谓的Visual C#就是指的可视化编程,主要在设计窗口布置好自己的控件(一些具有一定功能的小部件,例如如可以点击的按 ...

  10. Android 入门第一课 一个简单的提示框

    1.打开Android开发环境Eclipse来到主界面 2.新建一个安卓项目 File->New->Android Application project 在上面有红色错误的地方填上应用程 ...

随机推荐

  1. odoo关于计算字段store=True时导致的安装/更新时间较长问题的解决方案

    Odoo安装/更新模块原理 Odoo每次安装/更新模块时,会进行以下几步处理: 1.判断是否需要创建表,如果需要创建且表不存在,则进行表的创建(不进行字段的创建): 2.获取该表中已经存在的字段: 3 ...

  2. python 异步写入文件

    # -*- coding:utf-8 -*-import asyncioimport aiofilesimport time#异步操作时,函数名前必须加上asyncasync def func1(): ...

  3. DID 2022-12-02

    DID第二节课 最低工资对就业率影响(DID) 宾夕法尼亚VS新泽西的快餐店,用边界的好处:在用最低工资法之前,两地工资情况差不多,DID需要两期,92年11月实施,之前之后做电话访问. 数据越全的估 ...

  4. 2022年7月12,第四组,周鹏,被算法折磨的一天【哭】【哭】【哭】【puls哭】

    今天学习了JS的几种循环语法,说实话,前几天的简单让我大意了,没有闪,很成功的被搞崩了! 一杯水,一根烟,一个算法边写边骂是一天. 多少次,我满怀期待的以为它会出现想要的结果, 但现实的残酷狠狠的折磨 ...

  5. 一个实现单线程/多线程下代码调用链中传递数据的处理类: CallContext(LogicalSetData,LogicalGetData),含.net core的实现

    详情请参考原文:一个实现单线程/多线程下代码调用链中传递数据的处理类: CallContext

  6. DevExpress 的LayoutControl控件导致资源无法释放的问题处理

    现象记录 前段时间同事发现我们的软件在加载指定的插件界面后,关闭后插件的界面资源不能释放, 资源管理器中不管内存,还是GDI对象等相关资源都不会下降. 问题代码 问题的代码大概如下. public v ...

  7. linux基础:1、linux简介、虚拟化软件的安装与配置、Xshell的安装与配置

    Linux 目录 Linux 一.linux简介 二.linux发展史 三.虚拟化技术 1.简介 2.虚拟化软件下载 3.重要名词解释 4.远程链接工具 一.linux简介 常见岗位 1.自动化运维 ...

  8. KMP 与 Z 函数

    \(\text{By DaiRuiChen007}\) 一.KMP 算法 I. 问题描述 在文本串 \(S\) 中找到模式串 \(T\) 的所有出现,其中 \(|S|=n,|T|=m\) II. 前置 ...

  9. Flutter帧率监控 | 由浅入深,详解获取帧率的那些事

    前言 做线上帧率监控上报时,少不了需要弄明白如何通过代码获取实时帧率的需求,这篇文章通过图解配合Flutter性能调试工具的方式一步步通俗易懂地让你明白获取帧率的基础知识,以后再也不愁看不懂调试工具上 ...

  10. flutter2.x报错解决type (RouteSettings) => Route<dynamic> is not a subtype of type (RouteSettings) => Route<dynemic> of function result

    flutter2.x报错解决type (RouteSettings) => Route <dynamic>? is not a subtype of type (RouteSetti ...