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. nuxt作为主应用接入qiankun的实践(附代码)

    上半年一直在倒腾qiankun,在使用nuxtjs接入qiankun时遇到了一些坑,记录并分享出来,希望能帮助到大家. 代码地址:nuxtjs-qiankun-demo Nuxtjs接入qiankun ...

  2. 【Hadoop面试】基础概念、HDFS、MapReduce、Yarn、实战

    一.Hadoop概念及架构 1.是否看过Hadoop源码 2.正常工作的hadoop集群中hadoop都分别需要启动哪些进程,他们的作用分别是什么 3.hadoop和spark中的文件缓存方式 4.h ...

  3. C++日期和时间编程总结

    一,概述 二,C-style 日期和时间库 2.1,数据类型 2.2,函数 2.3,数据类型与函数关系梳理 2.4,时间类型 2.4.1,UTC 时间 2.4.2,本地时间 2.4.3,纪元时间 2. ...

  4. 云原生 • Kubernetes 认识 k8s、k8s 架构、核心概念点介绍

    云原生 • Kubernetes 认识 k8s.k8s 架构.核心概念点介绍 一.Kubernetes 简介Kubernetes 简称 k8s,是支持云原生部署的一个平台,起源于谷歌.谷歌早在十几年之 ...

  5. linux基础第二部分

    一.Linux命令执行过程 先判断是否是别名,如果是直接执行,不是看是否是内部命令 如果是内部命令,直接执行,不是看hash表 hash表中有源文件直接执行,找不到报错 若hash表中不存在去外部规定 ...

  6. Flaks框架(Flask请求响应,session,闪现,请求扩展,中间件,蓝图)

    目录 一:Flask请求响应 1.请求相关信息 2.flask新手四件套 3.响应相关信息(响应response增加数据返回) 二:session 1.session与cookie简介 2.在使用se ...

  7. SQL语句使用

    目录 一:sql语句 1.什么是SQL语句? 二:基本SQL语句之库操作 三:基本SQL语句之表操作 1.查看当前所在库名称 2.切换数据库 四:基本SQL语句之记录操作 五:创建表的完整语法 六:操 ...

  8. python运算符与基本数据类型

    Python种类 JavaPython cPython ***** pypy 字节码 和 机器码 Python程序: 1. 终端: C:\python35\python.exe D:\1.py 解释器 ...

  9. 【深入浅出 Yarn 架构与实现】4-2 RM 管理 Application Master

    上一篇文章对 ResourceManager 整体架构和功能进行了讲述.本篇将对 RM 中管理 Application Master 的部分进行深入的讲解. 下面将会介绍 RM 与 AM 整体通信执行 ...

  10. 自研ORM框架实现工作单元模式

    平时我们使用事务,需要显示的Try Catch 并且开启事务 提交事务 异常回滚事务 三步骤,使用工作单元后则只需要Commit. 1.接口定义 1 /// <summary> 2 /// ...