make 和 makefile 的关系
程序的 编译 和 链接
要先总结 make 和 makefile,就需要先了解下面这个过程:

- 预编译:也叫预处理,进行一些文本替换工作,比如将
#define定义的内容,在代码中进行替换; - 编译:将预处理得到的代码,进行词法分析、语法分析、中间代码……;如果是在Windows下,中间代码就是
.obj文件;在Linux系统下,中间代码就是.o文件; - 汇编:将编译得到的汇编代码,通过汇编程序得到 0 和 1 机器语言;
- 链接:链接各种静态链接库和动态链接库得到可执行文件。
make 和 makefile 能干啥?
一个工程,那么多源文件,一堆的 cpp 和 h 文件,怎么编译啊?编译一个大型工程,如果Rebuild可能就需要好几个小时,甚至十几个小时,那我们就可能要问了。
- 如何像VS那样,一键就能编译整个项目?
- 如何修改了哪个文件,就编译修改的那个文件,而不是重新编译整个工程?
好吧,make 和 makefile 就能搞定这些。makefile 定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后编译,哪些文件需要重新编译,甚至于进行更复杂的功能操作,因为 makefile 就像一个Shell脚本一样,其中也可以执行操作系统的命令。makefile 带来的好处就是——“自动化编译”,一旦写好,只需要一个 make 命令,整个工程完全自动编译,极大的提高了软件开发的效率。
make 是一个命令工具,是一个解释 makefile 中指令的命令工具,一般来说,大多数的IDE都有这个命令,比如:Delphi的make,Visual C++的nmake,Linux 下 GNU 的 make。可见,makefile 都成为了一种在工程方面的编译方法。make 命令执行时,需要一个 makefile 文件,以告诉 make 命令需要怎么样的去编译和链接程序。
现在,应该明白了吧。make 是一个命令,用来解析 makefile 文件;makefile 是一个文件,用来告诉 make 命令,如何编译整个工程,生成可执行文件。再打个比方:
导演 == make
剧本 == makefile
演员 == MAKE调用的外部命令,如编译器、链接器等
电影 == 生成的程序
解决问题举例
怎么就出现了make这个东西了呢?还记得你入门C语言时,写下的Hello World程序么?
#include <stdio.h>
int main()
{
printf("Hello World\n");
return 0;
}
当你在终端中输入 gcc HelloWorld.c 命令时,就会生成一个 a.out 文件(如果不用 -o 参数指定输出文件名的话,默认为 a.out),然后就可以神奇的使用 ./a.out 执行该文件,打印出了 Hello World。这是一件让初学者兴奋的事情。问题来了,现在就仅仅是一个 HelloWorld.c 文件,如果有多个代码文件,而多个代码文件之间又存在引用关系,这个时候,该如何去编译生成一个可执行文件呢?比如现在有一下源文件:
add.h
add.c
sub.h
sub.c
mul.h
mul.c
divi.h
divi.c
main.c
这些代码文件的定义分别如下:
add.h 文件
#ifndef _ADD_H_
#define _ADD_H_
int add(int a, int b);
#endif
add.c 文件
#include "add.h"
int add(int a, int b)
{
return a + b;
}
sub.h 文件
#ifndef _SUB_H_
#define _SUB_H_
int sub(int a, int b);
#endif
sub.c 文件
#include "sub.h"
int sub(int a, int b
{
return a - b;
}
mul.h 文件
#ifndef _MUL_H_
#define _MUL_H_
int mul(int a, int b);
#endif
mul.c 文件
#include "mul.h"
int mul(int a, int b)
{
return a * b;
}
divi.h 文件
#ifndef _DIVI_H_
#define _DIVI_H_
int divi(int a, int b);
#endif
divi.c 文件
#include "divi.h"
int divi(int a, int b)
{
if (b == 0)
{
return 0;
}
return a / b;
}
main.c 文件
#include <stdio.h>
#include "add.h"
#include "sub.h"
#include "mul.h"
#include "divi.h" int main()
{
int a = 10;
int b = 2; printf("%d + %d = %d\n", a, b, add(a, b));
printf("%d - %d = %d\n", a, b, sub(a, b));
printf("%d * %d = %d\n", a, b, mul(a, b));
printf("%d / %d = %d\n", a, b, divi(a, b));
return 0;
}
你也看到了,在 main.c 中要引用这些文件,那现在如何编译,生成一个可执行文件呢?
最笨的解决方法
最笨的解决方法就是依次编译所有文件,生成对应的 .o 目标文件。参考如下:
$ gcc -c sub.c -o sub.o
$ gcc -c add.c -o add.o
$ gcc -c sub.c -o sub.o
$ gcc -c mul.c -o mul.o
$ gcc -c divi.c -o divi.o
$ gcc -c main.c -o main.o
然后再使用如下命令对所生成的单个目标文件进行链接,生成可执行文件。
$ gcc -o main add.o sub.o mul.o divi.o main.o
然后就可以得到一个可执行程序 main,可以直接使用 ./main 进行运行。还不错,虽然过程艰辛,至少也可以得到可执行程序。那么有没有比这更简单的方法呢?如果一个项目,几千个文件,这么写下去,还不得累死人啊。办法是有的,我接着总结。
使用makefile文件
使用上面那种最笨的办法,效率是非常低得,当添加新的文件,或者修改现有文件时,维护起来也是非常难得。基于此,现在就来说说使用 makefile 文件来搞定这一切。
关于什么是 makefile,在文章的开头我就已经总结了,至于它和 make 的关系,在文章的开头也说的非常清楚了,现在就来看看如何使用 makefile 来完成上面同样的任务,生成一个 main 的可执行文件。
#target:dependency-file
main:main.o add.o sub.o mul.o divi.o
gcc -o main main.o add.o sub.o mul.o divi.o
main.o:main.c add.h sub.h mul.h divi.h
gcc -c main.c -o main.o
add.o:add.c add.h
gcc -c add.c -o add.o
sub.o:sub.c sub.h
gcc -c sub.c -o sub.o
mul.o:mul.c mul.h
gcc -c mul.c -o mul.o
divi.o:divi.c divi.h
gcc -c divi.c -o divi.o
clean:
rm -f *.o
上面就是 makefile 文件的内容,对于 makefile 的内容的编写规则,这里先不说。
现在你可以在 makefile 的同目录下执行 make 命令,然后就可以看到生成了一堆 .o 目标文件,还有那个可执行的 main 文件;接着运行 make clean,那些 .o 文件就全部被删除了。为什么是这样?好了,你先照着做一遍吧。
makefile文件编写规则
上面只是给出了一个简单的 makefile 文件,你肯定好奇这个 makefile 的书写规则是什么样子的?
makefile 的规则大体上就是以下格式:
target 是一个目标文件,可以是 Object File(.o文件),也可以使最终的执行文件,而 dependency-file 是生成对应 target 所需要依赖的文件或者其它的 target,command 就是最终由 make 执行的命令。
上面说了一段话,简短而言就是:生成一个 target,需要依赖的文件,而使用命令来将依赖文件生成对应的 target 的规则,是在 command 中定义的。如果 dependency-file 中有一个或者多个文件比 target 文件要新的话,command 所定义的命令就会被执行,这就是 makefile 的规则,也就是 makefile 最核心的内容。
makefile 文件中可以定义变量,可以使用函数,还有各种判断,内容繁多,这里就不一一总结了,更详细的介绍,可以看看大牛陈皓的系列博客《跟我一起写makefile》。
参考:
http://www.jellythink.com/archives/810?utm_source=tuicool&utm_medium=referral
http://bbs.csdn.net/topics/390143962
make 和 makefile 的关系的更多相关文章
- Makefile.am, Makefile.in 与 Makefile的关系(转)
文章出处:http://blog.mcuol.com/User/wangguangdong/Article/17384_1.htm Makefile.am, Makefile.in, Makefile ...
- Makefile依赖关系中的竖线“|”
网上搜索无果,于是自己查看了一下makefile的info文件,其中解释如下: [java] view plain copy print? target : prerequisites [TAB] ...
- ubuntu——Kconfig、.config、Makefile的关系
原文地址:http://blog.csdn.net/estate66/article/details/5886816 ,本人对此文有改进. 当我们编写完一个驱动后,我们要把它以模块形式编译或者直接编译 ...
- kernel Makefile Kconfig说明
实际文档位置:Documentation/kbuild/makefiles.txt,此为翻译稿. *************************************************** ...
- linux-2.6.22.6内核启动分析之Makefile文件
学习目标 分析Makefile文件,了解内核中的哪些文件被编译,如何被编译,连接时顺序如何确定! Linux内核源码中包含很多的Makefile文件,这些Makefile文件又包含其它的一些文件,比如 ...
- Makefile基础(一)
在大型的C语言项目中,一般都是由多个源文件编译链接形成的可执行程序,而这些源文件的处理步骤,通常交给Makefile来管理,Makefile定义了一系列的规则来指定,哪些文件需要先编译,哪些文件需要后 ...
- Make 和 Makefile快速入门
前言 一个项目,拥有成百上千的源程序文件,编译链接这些源文件都是有规则的.Makefile是整个工程的编译规则集合,只需要一个make命令,就可以实现“自动化编译”.make是一个解释makefile ...
- Linux C 收藏
某招聘要求:熟悉高性能分布式网络服务端设计开发,熟悉epoll.多线程.异步IO.事件驱动等服务端技术: <UNIX环境高级编程(第3版)>apue.h等源码文件的编译安装 <UNI ...
- gcc编译的四个阶段:预处理,编译,汇编,链接
1:gcc编译的四个阶段:预处理,编译,汇编,链接 #vi file.c #gcc -E file.c -o file.i//-E查看且预处理后停止编译,-o生成目标文件,-i表示已预处理 #gcc ...
随机推荐
- 深入理解Java反射
要想理解反射的原理,首先要了解什么是类型信息.Java让我们在运行时识别对象和类的信息,主要有2种方式:一种是传统的RTTI,它假定我们在编译时已经知道了所有的类型信息:另一种是反射机制,它允许我们在 ...
- noip2010提高组3题题解 by rLq
本题地址http://www.luogu.org/problem/show?pid=1525 关押罪犯 题目描述 S 城现有两座监狱,一共关押着N 名罪犯,编号分别为1~N.他们之间的关系自然也极不和 ...
- PHP用法
链接: php编写app接口(一)-JSON方式封装接口数据方法 php 非常有用的高级函数PATH_SEPARATOR常量和set_include_path date_default_timezon ...
- java多线程系类:JUC集合:01之框架
概要 之前,在"Java 集合系列目录(Category)"中,讲解了Java集合包中的各个类.接下来,将展开对JUC包中的集合进行学习.在学习之前,先温习一下"Java ...
- java并发编程学习: 原子变量(CAS)
先上一段代码: package test; public class Program { public static int i = 0; private static class Next exte ...
- 手机网页Html代码实现(解决显示页面很小的问题)
工作需要,要做一个手机自适应的网页效果,终于搞定,先分享并记录! 其实主要就是改掉HTML页面声明: 在网页中加入以下代码,就可以正常显示了: <meta name="viewport ...
- 2015年3月阿里内推(c++研发)实习生电面经历
2015年3月开学开始就听说阿里会有内推,果不其然在师兄的引荐下推了菜鸟网络,但是在学校的BBS上看到了阿里云部门,而且要会C++,这使我更感兴趣,重新选择了方向,当然最后选择了阿里云.在此分享一下三 ...
- 支持断线重连、永久watcher、递归操作并且能跨平台(.NET Core)的ZooKeeper异步客户端
在公司内部的微服务架构中有使用到了"ZooKeeper",虽然官方有提供了.NET的SDK,但易用性非常的差,且搜遍github.nuget,没有发现一个可以跨平台且易用的组件,所 ...
- session过期返回登录页面跳出frame
session 过期返回登录页面 方法1, HttpSession session = request.getSession(); String LOGIN_ID = (String) session ...
- 通过CAGradientLayer制作渐变色效果(转)
转载自:http://blog.it985.com/7986.html 看了极客学院的视频之后写的一篇博客,觉得不错,还是作为笔记使用. 简单介绍一下CAGradientLayer吧. Gradien ...