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. linux 挂载 vdi 文件(virtual box虚拟机镜像文件)

    1. 下载 vdfuse 下载地址 2.解压deb文件 解压deb安装包文件,这里不使用安装命令是因为你的virtualbox 可能和vdfuse的版本不一致,导致安装失败,而我们只需要用到 vdfu ...

  2. day31-JQuery04

    JQuery04 6.jQuery的DOM操作02 6.9常用遍历节点方法 取得匹配元素的所有子元素组成的集合:children(),该方法只考虑子元素而不考虑任何后代元素 取得匹配元素后面的同辈元素 ...

  3. JavaEE Day11 BootStrap

    之前:前端知识 HTML+CSS+JavaScript           不好写 今日内容:前端的开发框架,内部定义了丰富的CSS样式和JS代码,只需要拿来用就行 会用即可 一.BootStrap ...

  4. 【每日一题】【集合增删】2022年1月13日-NC41 最长无重复子数组-220113/220122

    描述 给定一个长度为n的数组arr,返回arr的最长无重复元素子数组的长度,无重复指的是所有数字都不相同. 子数组是连续的,比如[1,3,5,7,9]的子数组有[1,3],[3,5,7]等等,但是[1 ...

  5. DeprecationWarning: collection.ensureIndex is deprecated. Use createIndexes instead

    // 引入mongoose模块 const mongoose = require('mongoose'); // 链接数据库 mongoose.set('useCreateIndex', true)  ...

  6. Selenium4+Python3系列(十三) - 与docker中的jenkins持续集成

    前言 文章更新到这一篇时,其实我还是很开心的,因为这也正是这系列教程的最后一篇文章,也算是完成了一个阶段性的小目标,也很感谢那些愿意看我文章与我交流学习的同学,感谢有你们的支持和陪伴. Jenkins ...

  7. Django框架路由层-无名有名分组-无名有名分组反向解析

    目录 一:路由层 1.路由匹配(错误演示) 2.路由匹配错误原因 3.路由匹配(解决方式1) 4.settings配置文件控制自动添加斜杠匹配 5.url方法第一个参数是正则表达式(正规使用url) ...

  8. 小型web产品的功能测试要点或测试大纲

    本文参考配置啦:-- Web类产品功能测试大纲,黑盒测试参考测试范围 [官网]:无 应用场景 黑盒测试,功能测试中常常需要考虑很多问题,这里根据本人的工作经验遇到的进行了系列总结.给出了一个常用的测试 ...

  9. CFS三层内网靶场

    前言 最近学习了内网的一些知识,想着打一下靶场来试试,选择了这个CFS的三层内网靶场,做一下记录 靶场下载地址   链接:https://pan.baidu.com/s/1zGw5VNt222nCmf ...

  10. HttpServletRequest获取参数和文件

    从HttpServletRequest中获取上传文件及参数的方法 try { //下面这句必须加,不然报错 MultipartResolver resolver = new CommonsMultip ...