make工具与Makefile文件
make工具与Makefile文件
阅读目录
正文
1. make工具
利用make工具可以自动完成编译工作,这些工作包括:
- 如果修改了某几个源文件,则只重新编译这几个源文件
- 如果某个头文件被修改了,则重新编译所有包含该头文件的源文件
利用这种自动编译可以大大简化开发工作,避免不必要的重新编译。make工具通过一个称为Makefile的文件来完成并自动维护编译工作,Makefile文件描述了整个工程的编译、连接规则。
2. Makefile文件
Makefile描述了整个工程的编译连接规则。Makefile的基本规则为:
TARGET...: DEPENDENCIES...
COMMAND
...
- TARGER:目标程序产生的文件,如可执行文件和目标文件,目标也可以是要执行的动作,如clean,也称为伪目标。
- DEPENDENCIES:依赖是用来产生目标的输入文件列表,一个目标通常依赖与多个文件。
- COMMAND:命令是make执行的动作(命令是shell命令或是可在shell下执行的程序),注意每个命令行的起始字符必须为TAB字符。
- 如果DEPENDENCIES中有一个或多个文件更新的话,COMMAND就要执行,这就是Makefile最核心的内容。
3. Makefile的简单示例
$ touch add.c add.h sub.c sub.h main.c
现在有这5个文件add.h 、sub.h中包含了函数声明,add.c、sub.c中包含了函数实现,main.c调用了函数。Makefile的文件:
main:main.o add.o sub.o 【目标文件是main,它依赖于main.o,add.o,sub.o这三个文件】
gcc -Wall -g main.o add.o sub.o -o main 【由依赖文件生成目标文件应该执行的命令】
main.o:main.c
gcc -Wall -g -c main.c -o main.o
add.o:add.c add.h
gcc -Wall -g -c add.c -o add.o
sub.o:sub.c sub.h
gcc -Wall -g -c sub.c -o sub.o
保存Makefile文件后执行make命令:
$ make
gcc -Wall -g -c main.c -o main.o
gcc -Wall -g -c add.c -o add.o
gcc -Wall -g -c sub.c -o sub.o
gcc -Wall -g main.o add.o sub.o -o main
可以看到执行了make之后,由于 目标文件main依赖于 main.o add.o sub.o ,所以是需要先 生成 这三个.o文件,最后才生成main。
如果此时再次输入make,会看到:
$ make
make: 'main' is up to date.
make的编译规则是根据时间来进行判断,一旦依赖列表中某个文件的更新时间比目标文件晚,则会重新生成目标,否则会出现以上提示。
默认情况下敲击make将生成第一个目标,也就是main。也可以生成指定的目标:
$ make add.o 【指定只生成add.o文件】
Makefile文件的名字不一定得命名为“Makefile”或"makefile",使用其他名字也是可以的。例如我们由一个文件叫myMakefile,同样可以使用它:
make -f myMakefile 【-f 选项的作用是把名字"myMakefile"作为makefile来对待。】
4. 伪目标
TARGET...: DEPENDENCIES...
COMMAND 【注意COMMAND之前是一个TAB,不是空格】
...
前面说过,TARGET除了可以是目标文件之外,还可以是伪目标。执行伪目标的效果等于执行了某一个动作, 并不产生目标文件。例如添加一个伪目标:
main:main.o add.o sub.o
gcc -Wall -g main.o add.o sub.o -o main
main.o:main.c
gcc -Wall -g -c main.c -o main.o
add.o:add.c add.h
gcc -Wall -g -c add.c -o add.o
sub.o:sub.c sub.h
gcc -Wall -g -c sub.c -o sub.o
clean : 【这是一个伪目标】
rm -f $(OBJECTS) main
使用make来执行伪目标:
$ make clean
rm -f main.o add.o sub.o main
可以看到make将执行伪目标下面的命令。
5. Makefile 自动化变量
从上面的Makefile文件我们发现一些问题:有时候目标文件的依赖列表过长,或者命令重复书写。利用Makefile自动化变量可以解决这个问题。
选项名 | 作用 |
---|---|
$@ | 规则的目标文件名 |
$< | 规则的第一个依赖文件名 |
$^ | 规则的所有依赖文件列表 |
刚才的Makefile文件,我们可以改写为:
main:main.o add.o sub.o
gcc -Wall -g $^ -o $@ 【等价于 gcc -Wall -g main.o add.o sub.o -o main】
main.o:main.c
gcc -Wall -g -c $< -o $@
add.o:add.c add.h
gcc -Wall -g -c $< -o $@
sub.o:sub.c sub.h
gcc -Wall -g -c $< -o $@
执行make,可以看到效果和之前是一样的:
$ make
gcc -Wall -g -c main.c -o main.o
gcc -Wall -g -c add.c -o add.o
gcc -Wall -g -c sub.c -o sub.o
gcc -Wall -g main.o add.o sub.o -o main
还可以自定义变量:
OBJECTS = main.o add .o sub.o 【OBJECTS是自定义的变量名】
main:$(OBJECTS) 【可以在需要的地方使用变量名进行替换,替换规则为$(变量名)】
gcc -Wall -g $^ -o $@
main.o:main.c
gcc -Wall -g -c $< -o $@
add.o:add.c add.h
gcc -Wall -g -c $< -o $@
sub.o:sub.c sub.h
gcc -Wall -g -c $< -o $@
6. 编译生成多个可执行文件
假设现在不只是想生成可执行main,还想生成可执行文件main2,可以这样写
BIN = main main2 【自定义变量BIN】
OBJECTS= main.o add.o sub.o
all : $(BIN) 【关注重点】
main : $(OBJECTS)
gcc -Wall -g $< -o $@
main2: $(OBJECTS)
gcc -Wall -g $< -o $@
main.o : main.c
gcc -Wall -g -c $< -o $@
main2.o :msin2.c
gcc -Wall -g -c $< -o $@
add.o:add.c add.h
gcc -Wall -g -c $< -o $@
sub.o:sub.c sub.h
gcc -Wall -g -c $< -o $@
clean :
rm -f $(OBJECTS) $(BIN)
为了生成目标文件all,需要先生成BIN,也即是 main main2。这样就可以生成两个可执行文件了。利用自定义变量可以再简化这段Makefile文件:
BIN = main main2
OBJECTS= main.o add.o sub.o
CC = gcc
CFALGS = -Wall -g
all : $(BIN)
main : $(OBJECTS)
$(CC) $(CFALGS) $< -o $@
main2: $(OBJECTS)
$(CC) $(CFALGS) $< -o $@
main.o : main.c
$(CC) $(CFALGS) -c $< -o $@
main2.o :msin2.c
$(CC) $(CFALGS) -c $< -o $@
add.o:add.c add.h
$(CC) $(CFALGS) -c $< -o $@
sub.o:sub.c sub.h
$(CC) $(CFALGS) -c $< -o $@
clean :
rm -f $(OBJECTS) $(BIN)
但是这样看起来,重复的内容还是比较多,可以使用下面的方法来继续简化:
BIN = main main2
OBJECTS= main.o add.o sub.o
CC = gcc
CFALGS = -Wall -g
all : $(BIN)
main : $(OBJECTS)
$(CC) $(CFALGS) $< -o $@
main2: $(OBJECTS)
$(CC) $(CFALGS) $< -o $@
.o .c : 【关注重点在这里】
$(CC) $(CFALGS) -c $< -o $@
clean :
rm -f $(OBJECTS) $(BIN)
利用 .o.c :,可以自动地把所有的.c文件到.o文件的生成都使用同一条命令来完成,简化的重复的工作。
7. make常用的内嵌函数
首先看make中函数调用的形式:
//函数调用
$(function arguments) 【function是函数名称,arguments是参数,使用$来调用】
值得注意的是,函数名称与参数之间是空格。
来看三个常用make内嵌函数。
- $(wildcard PATTERN) 作用是在当前目录下匹配模式的文件。
src = $(wildcard *.c) 【在当前目录下搜索所有.c文件,文件名称列表保存到src中】
- $(patsubst PATTENR,REPLACEMENT,TEXT) 模式替换函数,作用是把TEXT中文件列表从模式PATTENR替换为REPLACEMENT模式。
$(patsubst %.c,%.o,$src) 【把src中的.c文件列表中的文件从.c替换为.o】
等价于:
$(src:.c =.o) 【这种方式更常用】
- shell函数
shell函数可以执行shell下的命令,同样是使用$来引用,例如
$(shell ls -d */) 【将当前目录下的所有文件夹都列出来】
下面通过一个多级目录的例子来使用这些函数。场景是这样的,当前目录下有main.c文件,同时还有若干个目录,每个目录中都有各自的.c文件。利用所有的.c文件编译生成最后的main文件:
CC = gcc
CFLAGS = -Wall -g
BIN = main
SUBDIR = $(shell ls -d */) 【SUBDIR变量保存了子目录的列表】
ROOTSRC = $(wildcard *.c) 【ROOTSRC保存了当前目录下的.c文件列表】
ROOTOBJ = $(ROOTSRC:%.c = %.o) 【ROOTBOJ 保存了当前目录下.c文件同名的.o列表】
SUBSRC = $(shell find $(SUBDIR) -name '*.c') 【SUBSRC 保存了所有子目录下的的.c文件】
SUBOBJ = $(SUBSRC:%.c = %.o) 【SUBOBJ保存了所有子目录下的.c文件同名的.o文件列表】
$(BIN):$(ROOTOBJ) $(SUBOBJ) 【main的生成依赖与当前目录及所有子目录下的.o文件】
$(CC) $(CFLAGS) -o $(BIN) $(ROOTOBJ) $(SUBOBJ)
.o .c:
$(CC) $(CFLAGS) -c $< -o $@
clean:
rm -f $(BIN) $(ROOTOBJ) $(SUBOBJ)
文章链接:http://www.cnblogs.com/QG-whz/p/5461110.html
make工具与Makefile文件的更多相关文章
- Linux工具入门:make工具与Makefile文件
1. make工具 利用make工具可以自动完成编译工作,这些工作包括: 如果修改了某几个源文件,则只重新编译这几个源文件 如果某个头文件被修改了,则重新编译所有包含该头文件的源文件 利用这种自动编译 ...
- 使用autoconf和automake生成Makefile文件(转)
Makefile好难写 曾经也总结了一篇关于Makefile的文章<make和makefile的简单学习>.但是,总结完以后,发现写Makefile真的是一件非常痛苦的事情,的确非常痛苦. ...
- 面试题----makefile文件的作用
make工具和makefile文件 make工具和makefile文件简介 make命令和makefile文件的结合提供了一个在项目管理领域十分强大的工具.它不仅常被用于控制源代码的编译和链接,而且还 ...
- linux下使用automake工具自动生成makefile文件
linux环境下,当项目工程很大的时候,编译的过程很复杂,所以需要使用make工具,自动进行编译安装,但是手写makefile文件比较复杂,所幸在GNU的计划中,设计出了一种叫做Autoconf/Au ...
- 编写一个通用的Makefile文件
1.1在这之前,我们需要了解程序的编译过程 a.预处理:检查语法错误,展开宏,包含头文件等 b.编译:*.c-->*.S c.汇编:*.S-->*.o d.链接:.o +库文件=*.exe ...
- Makefile文件学习总结
Makefile文件相当于是一种脚本编程语言,目的是实现自动化编译.编写makefile文件的过程中可以使用变量.控制结构和函数等一般编程语言的特性. Makefile文件的组成内容.makefile ...
- 利用 autoconf 和 automake 生成 Makefile 文件
一.相关概念的介绍 什么是 Makefile?怎么书写 Makefile?竟然有工具可以自动生成 Makefile?怎么生成啊?开始的时候,我有这么多疑问,所以,必须得先把基本的概念搞个清楚. 1.M ...
- linux中Makefile文件相关内容
第一章.概述什么是makefile?或许很多Winodws的程序员都不知道这个东西,因为那些Windows的IDE都为你做了这个工作,但我觉得要作一个好的和professional(专业)的程序员,m ...
- 使用automake等命令自动生成Makefile文件 (转载)
使用automake等命令自动生成Makefile文件 Linux下编程时,为了方便编译,往往使用Makefile文件自动完成编译,但是Makefile文件本身的书写十分复杂,规则很多.好在Lin ...
随机推荐
- PHP - 使用 Pear 进行安装和卸载包
安装: 首先运行到php根目录: 输入要安装的包文件名: 使用语法: pear install 要安装包的名称 回车确认: 如果没有其他意外,显示安装成功. 查看安装的包的信息: 语句: pear i ...
- HDU 1425 sort 题解
选择出数列中前k个最大的数. 这里由于数据特殊.所以能够使用hash表的方法: #include <cstdio> #include <algorithm> #include ...
- Python Challenge 第四题
这一题没有显示提示语,仅仅有一幅图片,图片也看不出什么名堂,于是直接查看源代码,源代码例如以下: <html> <head> <title>follow the c ...
- asp.net下利用MVC模式实现Extjs表格增删改查
在网上看到有很多人写extjs下的表格控件的增删改查,但是大多数都是直接从后台读取数据,很少有跟数据库进行交互的模式. 今天就来写一个这样的例子.欢迎大家交流指正. 首先简单介绍一下MVC模式,MVC ...
- Spring Mobile是如何判断访问设备的类型的
Spring最近换域名了,去转转,发现了一个有意思的项目:spring mobile. http://projects.spring.io/spring-mobile/ 这个项目有很多实用的功能,如识 ...
- for_each的各种情况下的使用详解
原创作者:http://oomusou.cnblogs.com 配合<C++ Template>(简体中文)使用 http://download.csdn.net/detail/qq239 ...
- DM6446开发攻略——u-boot-1.3.4移植(1)
http://zjbintsystem.blog.51cto.com/964211/282387转载 UBOOT的版本更新速度比较快,截止今天,稳定正式的版本是u-boot-2009.11-rc2 ...
- Theano学习笔记(二)——逻辑回归函数解析
有了前面的准备,能够用Theano实现一个逻辑回归程序.逻辑回归是典型的有监督学习. 为了形象.这里我们如果分类任务是区分人与狗的照片. 首先是生成随机数对象 importnumpy importth ...
- NLP | 自然语言处理 - 标注问题与隐马尔科夫模型(Tagging Problems, and Hidden Markov Models)
什么是标注? 在自然语言处理中有一个常见的任务,即标注.常见的有:1)词性标注(Part-Of-Speech Tagging),将句子中的每一个词标注词性,比如名词.动词等:2)实体标注(Name E ...
- Swift - 设置网格UICollectionView的单元格间距
要设置单元格cell的间距(水平间距,垂直间距)可进行如下设置: 方法1:在storyboard中设置 选择Collection View后在面板里设置Min Spacing相关属性(这里也可以设置单 ...