MakeFile的规则
简单形式的规则:
target ... : prerequisites ...
command
...
...
...
结构 | 解释 |
---|---|
target |
可以是一个目标文件,也可以是执行文件或者一个标签 |
prerequisties |
生成target所需要依赖的文件和/ 或者target |
command |
target需要执行的命令 |
makefile其实是一个文件的依赖关系,也就是说,target依赖于prerequisites的文件,其生成规则定义在command中
prerequisites 中如果有文件比target文件更新的话,那么command中的命令就会被执行
简单示例
edit : main.o kbd.o command.o display.o
insert.o search.o files.o utils.o
cc -o edit main.o kdb.o command.o display.o
insert.o search.o files.o utils.o
main.o : main.c defs.h
cc -c main.c
kbd.o : kbd.c defs.h command.h
cc -c kbd.c
command.o : command.c defs.h buffer.h
cc -c command.c
display.o : display.c defs.h buffer.h
cc -c display.c
insert.o : insert.c defs.h buffer.h
cc -c insert.c
search.o : search.c defs.h buffer.h
cc -c search.c
files.o : files.c defs.h buffer.h command.h
cc -c files.c
utils.o : utils.c defs.h
cc -c utils.c
clean :
rm edit main.o kdb.o command.o display.o
insert.o search.o files.o utils.o
简单解释;
是表示换行的意思,文件名可以保存为
makefile
或者Makefile
,然后可以直接在该目录先输入make
变可以生成edit
了。 如果需要删除执行中生成的中间文件,那么可以执行make clean
。
在上面的makefile中,target是edit和其他中间目标文件,prerequisites就是:后面那些.c文件和.h文件。每个.o文件都有自己的依赖熟悉,每个.o文件又是edit的依赖属性。
依赖关系的实质就是定义了目标文件是由哪些文件生成的;
command必须以一个Tab
作为开头,不然会报错;
clean
其实是一个动作的名字,不是像之前的.o文件;
makefile的工作原理
make
会在当前目录下找名字叫Makefile
或者makefile
的文件;- 如果找到,会在文件中找一个
target
,示例程序中就是edit
,并且把它作为最终的目标文件; - 如果edit不存在,或者是edit后的依赖属性比edit还要新,那么就会执行定义的命令来生成edit;
- 如果edit后面的
.o
文件也不存在,那么make
会在文件中寻找target为.o的target,并且根据对应的规则来生成.o文件,前提是需要的.c
和.h
文件都存在,最后还是会生成edit;
makefile中使用变量
示例程序中在生成edit的时候有很多.o文件,这个时候很容易犯错误,则可以定义一个变量来代表这些,就像是C++中的宏定义一样,变量声明的示例程序如下:
OBJECTS = main.o kbd.o command.o display.o
insert.o search.o files.o utils.o
完整程序如下所示:
OBJECTS = main.o kbd.o command.o display.o
insert.o search.o files.o utils.o
edit : $(OBJECTS)
cc -o edit $(OBJECTS)
main.o : main.c defs.h
cc -c main.c
kbd.o : kbd.c defs.h command.h
cc -c kbd.c
command.o : command.c defs.h buffer.h
cc -c command.c
display.o : display.c defs.h buffer.h
cc -c display.c
insert.o : insert.c defs.h buffer.h
cc -c insert.c
search.o : search.c defs.h buffer.h
cc -c search.c
files.o : files.c defs.h buffer.h command.h
cc -c files.c
utils.o : utils.c defs.h
cc -c utils.c
clean :
rm edit $(OBJECTS)
makefile的自动推导
make
看到一个.o
文件,那么就会自动把.c
文件加到依赖关系
中,例如找到一个hello.o ,那么hello.c就是hello.o的依赖文件,并且cc -c hello.c
也会被推导出来
利用上面介绍的自动推导,则可以进一步的简化示例程序中的makefile了:
objects = main.o kbd.o command.o display.o
insert.o search.o files.o utils.o
edit : $(objects)
cc -o edit $(objects)
main.o : defs.h
kbd.o : defs.h command.h
command.o : defs.h command.h
display.o : defs.h buffer.h
insert.o : defs.h buffer.h
search.o : defs.h buffer.h
files.o : defs.h buffer.h command.h
utils.o : defs.h
.PHONY : clean
clean :
rm edit $(objects)
makefile的清理规则
每个makefile建议都写一个clean target的规则,这不仅利于重新编译,也利于保持文件清洁。
写法一:
clean:
rm edit $(objects)
写法二:
.PHONY : clean
clean:
-rm edit $(objects)
引用其他的makefile
需要使用关键字
include
,前面不能以TAB键开始,但是可以有一些空字符,格式如下include
可以用一个或者多个空格分开
例如,有这几个makefile,分别是:a.make, b.mk,c.mk 以及一个变量$(test),变量包含了d.mk和f.mk。那么,如果想要引用这些makefile,则可以使用下面的语法:
include amake *.mk $(test)
等价于
include a.make b.mk b.mk d.mk f.mk