makefile 学习(一)
一、Makefile的基本规则
GNU make 规则:
target ... : prerequisites ...
command
....
....
target —
目标文件, 可以是Object File 也可以是可执行文件,还可也是标签Label(标签内容在“伪目标”章节);
prerequisites—生成target所需的文件或目标;
command—make需要执行的命令,可以是任何shell命令。
二、一个简单的例子
创建一个名为count_word.c的文件,代码如下
- #include <stdio.h>
- extern int fee_count, fie_count, foe_count, fum_count;
- extern int yylex( void );
- int main( int argc, char ** argv ){
- yylex( );
- printf( "%d %d %d %d\n", fee_count, fie_count, foe_count, fum_count );
- return( 0 );
- }
另外创建一个lexer.l文件,其中所有的空白均为tab键
- int fee_count = 0;
- int fie_count = 0;
- int foe_count = 0;
- int fum_count = 0;
- %%
- fee fee_count++;
- fie fie_count++;
- foe foe_count++;
- fum fum_count++;
最后创建makefile文件,内容为:
- count_words: count_words.o lexer.o -lfl
- gcc count_words.o lexer.o -lfl -o count_words
- count_words.o: count_words.c
- gcc -c count_words.c
- lexer.o: lexer.c
- gcc -c lexer.c
- lexer.c: lexer.l
- flex -t lexer.l > lexer.c
- clean:
- rm lexer.c lexer.o count_words.o count_words
以上内容保存在Makefile或者是makefile都可以,直接输入make命令就可以生成可执行文件count_words了;
gcc -c count_words.c
flex -t lexer.l > lexer.c
gcc -c lexer.c
gcc count_words.o lexer.o -lfl -o count_words
如果要删除执行文件和中间的目标文件,那么就执行一下make
clean。
注意1: 当依赖关系定好后,下面一行就是如何生成目标文件的操作系统命令了,一定要以一个Tab键开头。 另外,make会比较targets文件和prerequisites文件的修改日期,如果prerequisites文件的日期比targets文件新,或者targets不存在,那么make就会执行这下面一行的系统命令。
注意2: clean不是一个文件,它是一个动作名,冒号后面什么都没有,make就不会自动去找它的依赖性,也不会执行它后面的系统命令。因此,要执行clean就需要显式的指出make clean。
注意3: 如果报错,可能需要先安装flex:
- sudo apt-get install flex
注意4: 运行count_works后,它会回显你的输出并统计'fee','fie','foe','fum'的次数。结束统计需要按Ctrl+d,然后会输出四个单词出现的次数。
三、make 如何工作
默认方式
直接输入make,则
- make会在当前的目录下找到名为“Makefile”或者“makefile”的文件。
- 如果找到,它会把文件中第一个target作为最终的目标文件(如上面例子中的count_words)。
- 首先,make会检查目标count_words的prerequisite文件count_words.o, lexer.o 和 -lfl。
- count_words.o通过编译count_words.c生成
- lexer.o通过编译lexer.c 生成,但是lexer.c 并不存在,因此会继续寻找lexer.c的生成方式,并找到了通过flex程序将lexer.l生成为lexer.c。
- 最后,make会检查-lfl,-l是gcc的一个命令选项,表示将系统库链接到程序。而"fl"对应的是libfl.a的库。(GNU make 可以识别这样的命令,当一个prerequisite是以这种-l<name>的形式表示出来的时候,make会自己搜索lib<name>.so的库文件,如果没找到则继续搜索lib<name>.a的库文件)。这里make找到的是/usr/lib/libfl.a文件,并将它与程序进行连接。
- 如果count_words文件不存在,或者count_words所依赖的后面的.o文件的修改时间比count_words本身更加新,那么,它会执行后面定义的命令来生成这个count_words文件。如果count_words所依赖的.o文件也不存在,那么make会继续按照前面的方式生成.o文件。
- 找到相应的.c和.h,用来生成.o,然后再用.o完成make的最终任务。
关于依赖关系
make会一层一层的去找文件的依赖关系,最终编译出第一个目标文件。
关于重新编译
只要任何prerequisite 比 target新,那么这个目标文件就会被下面的命令重新生成。每一个命令都会被传递到shell中,并在自己的子shell里面执行。
关于错误
如果在寻找过程中出现错误,如文件找不到,则make会直接退出并报错。对于所定义的命令错误或者编译不成功,make是不会理会的,它只负责文件的依赖性。
四、变量使用
上面的例子可以看到,文件或者目标的名字几乎都毫无例外的出现了至少两次,甚至如果算上clean的内容,有些文件名出现了三次。然而,在一个大型的工程中这种情况会更加复杂,任何不经意的错误都会导致编译失败。为了让makefile更容易维护,在makefile中我们可以使用变量,或者更确切的说是一个字符串,类似c语言中的宏。例如:
- CC = gcc
- object = lexer.o count_words.o
- count_words: $(object) -lfl
- $(CC) $(object) -lfl -o count_words
- count_words.o: count_words.c
- $(CC) -c count_words.c
- lexer.o: lexer.c
- $(CC) -c lexer.c
- lexer.c: lexer.l
- flex -t lexer.l > lexer.c
- clean:
- rm lexer.c $(object) count_words
五、自动推导依赖关系
GNU make可以根据.o文件的文件名自动推导出同名的.c文件并加入依赖关系,不需要我们手动注明。并且gcc -c也会被自动推导出来,于是我们的makefile就变成了
- CC = gcc
- object = lexer.o count_words.o
- count_words: $(object) -lfl
- $(CC) $(object) -lfl -o count_words
- count_words.o:
- lexer.o:
- lexer.c: lexer.l
- flex -t lexer.l > lexer.c
- clean:
- rm lexer.c $(object) count_words
这种方法也叫“隐式规则”。
六、关于Clean
一个好习惯是每个makefile都要写clean规则,这样不仅可以方便重编译,也有利于保持文件路径的清洁。一般的风格是:
- clean:
- rm lexer.c $(object) count_words
但是,更为稳妥的做法是:
- .PHONY: clean
- clean:
- -rm lexer.c $(object) count_words
.PHONY表示clean是一个“伪目标”,而rm命令前面的减号则表示,不管出现什么问题都要继续做后面的事情。
注意:clean规则不要放在makefile的开头,不然就会变成make的默认目标了。
makefile 学习(一)的更多相关文章
- [转]Windows平台下Makefile学习笔记
Windows平台下Makefile学习笔记(一) 作者:朱金灿 来源:http://blog.csdn.net/clever101 决心学习Makefile,一方面是为了解决编译开源代码时需要跨编译 ...
- makefile 学习归纳
makefile 学习归纳 一直希望 好好整理下 makefile的写法,这在linux编程界是必备技能.下面就好好的说道说道. 可以参考的大神总结 整理 makefile是供make命令执行的 脚本 ...
- <转>Windows平台下Makefile学习笔记(二)
本文转自:http://blog.csdn.net/clever101/article/details/8286066 上次我们学习了怎么用Makefile编译一个控制台工程.这次我们学习一下如何使用 ...
- makefile学习(1)
GNU Make / Makefile 学习资料 GNU Make学习总结(一) GNU Make学习总结(二) 这篇学习总结,从一个简单的小例子开始,逐步加深,来讲解Makefile的用法. 最后用 ...
- (二)我的Makefile学习冲动&&编译过程概述
前言 一 年轻的冲动 二 学习曲线 1 Makefile基本语法 2 bash基础 3 world 三 编译过程概述 1 主机预装工具 2 编译host工具 3 编译交叉工具链 4 编译内核模块 5 ...
- Makefile学习笔记
ls -l 查看文件详细信息 1.gcc -E test.c -o test.i//预编译gedit test.i //查看:高级C 2.gcc -Wall -S test.i -o test.s// ...
- makefile学习笔记(多目录嵌套调用、变量使用)
http://blog.csdn.net/leexiang_han/article/details/9274229 学习了几天的makefile的嵌套调用编写也有一些心得,先声明,我也是初学者写文 ...
- Makefile学习(一)变量
鉴于之前有一些了解,还有自己的学习习惯,我一上来就看Makefile的变量这一章.主要脉络是根据GNU make中文手册. 第六章:Makefile中的变量 6使用变量 定义:变量是一个名字,代表一个 ...
- makefile 学习一
近期在学习nginx,由于实在linux下,一些代码须要用makefile文件来编译,比較节省时间. 由于在nginx中加入一个新的模块假设用./configure方法来加入,特别是当你的代码有错时, ...
- Makefile学习总结
Makefile用法分析 在linux开发中,应用程序的编译基本都采用GNU的make工具,而make搭配Makefile来实现工程代码的编译,在越是大型复杂的项目中,make的强悍之处越是明 ...
随机推荐
- Tomcat的startup.bat一闪而过问题的解决
问题描述:点击Tomcat的startup.bat,一闪而过. 问题分析: 1.Tomcat的startup.bat--->catalina.bat--->setclasspath.bat ...
- [platform]新旧内核的device设备注册对比
转自:http://blog.chinaunix.net/uid-7332782-id-3268801.html 1. Version2.6内核启动过程 start_kernel( ) //板子上电启 ...
- android中的数据库操作
如何在android中调用数据库资源 在android中主要有两种方法来实现对数据库的访问,一种是adb shell方式,另一种是通过相关的android 的java类来间接的对数据库来进行操作.其中 ...
- Redis应用案例,查找某个值的范围(转)
本文来自Redis在Google Group上的一个问题,有一位同学发贴求助,说要解决如下的一个问题:他有一个IP范围对应地址的列表,现在需要给出一个IP的情况下,迅速的查找到这个IP在哪个范围,也就 ...
- Redis简介及3.0.2编译安装
由于项目需要Redis作为内存数据库,所以也开始搞Redis~ Redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).l ...
- CE_现金的利息设定和计算(案例)
2014-07-15 Created By BaoXinjian 一.摘要 存款利率指客户按照约定条件存入银行帐户的货币,一定时间内利息额同贷出金额即本金的利率.有活期利率和定期利率之分,有年/月/日 ...
- Javascript函数声明与函数表达式
在定义函数时,我们一般使用下面这两种方法: 使用函数声明定义: function sum (a, b) { return a + b; } 使用函数表达式定义: var sum = function ...
- 简单数据访问类,生成简单SQL,自动转换成java对象
import java.util.HashMap; import java.util.List; import java.util.Map; import org.slf4j.Logger; impo ...
- c语言知识(2)
while(n)首先,n在这里被当作了一个条件其次,当n为真时,进入while循环体(A):否则跳出循环继续执行下面部分(B).n为0时条件为假 n不为0条件为真 while(j) { if(j%1 ...
- Python从2.6升级到2.7,使用pip安装module,报错:No Module named pip.log(转载)
From:http://blog.csdn.net/iefreer/article/details/8086834 python升级后,使用pip安装module,错误: 错误原因:版本升级后,之前的 ...