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. Spring Boot 多数据源配置

    第一种方式: AbstractRoutingDataSource 1.1. 手动切换数据源 application.properties # Order # 如果用Druid作为数据源,应该用url属 ...

  2. 说一下 ArrayDeque 和 LinkedList 的区别?

    大家好,我是小彭. 在上一篇文章里,我们聊到了基于链表的 Queue 和 Stack 实现 -- LinkedList.那么 Java 中有没有基于数组的 Queue 和 Stack 实现呢?今天我们 ...

  3. springboot接收前端传参的几种方式

    1.通过HttpServletRequest接收,常用于获取请求头参数以及Cookie,适用于GET 和 POST请求方式,以下两种方式: @GetMapping("/demo1" ...

  4. 【Java SE进阶】Day05 异常,线程

    一.异常 1.概念 程序执行过程中,出现非正常情况导致JVM的非正常停止 本身是一个类,产生异常即创建并抛出一个异常对象 Java处理异常的方式是进行中断处理 异常非语法错误,语法错误直接不会产生cl ...

  5. 【Day03】Spring cloud:源码讲解与容器化初探

    今日内容 原理和源码 容器化过度 一.Naocs 1.介绍 server端 启动入口类(Spring Boot项目,提供8848端口的监听访问) 源码包含InstanceController类(ser ...

  6. 乐观锁思想在JAVA中的实现——CAS

    更多技术干活尽在个人公众号--JAVA旭阳 前言 生活中我们看待一个事物总有不同的态度,比如半瓶水,悲观的人会觉得只有半瓶水了,而乐观的人则会认为还有半瓶水呢.很多技术思想往往源于生活,因此在多个线程 ...

  7. 中国风?古典系?AI中文绘图创作尝鲜!⛵

    作者:韩信子@ShowMeAI 深度学习实战系列:https://www.showmeai.tech/tutorials/42 本文地址:https://www.showmeai.tech/artic ...

  8. [python] tensorflow中的argmax()函数argmax()函数

    首先 import tensorflow as tf tf.argmax(tenso,n)函数会返回tensor中参数指定的维度中的最大值的索引或者向量.当tensor为矩阵返回向量,tensor为向 ...

  9. 计算1+2!+3!+...+n!的和

    计算1+2!+3!+...+n!的和 Code 点击查看代码 #include<iostream> #include<vector> using namespace std; ...

  10. 腾讯微信开源数据库PhxSQL简单部署记录

    1.建立文件夹与互信关系 [root@mysql-100 ~]# mkdir -p /app/soft/phxsql [root@mysql-100 phxsql]# ssh-keygen -t rs ...