实例代码:
#include <stdio.h>
int main()
{
printf("hello Makefile!\n");
return 0;
}
Makefile文件
myproc:myproc.c
gcc -o myproc myproc.c
.PHONY:clean
clean:
rm -f myproc
解释一下:
依赖关系:上⾯的⽂件myproc,它依赖myproc.c
依赖方法:gcc -o myproc myproc.c ,就是与之对应的依赖关系
项目清理:
伪目标文件总是被执行的。 什么叫做总是被执⾏?
查看状态, 我们可以看到以下三个时间。 文件 = 内容 + 属性 以下简称acm时间, a表示最近一次访问的时间, M表示内容最近被修改的时间, C表示最近一个属性被修改的时间。 例如 : cat proc.c就是访问文件(但是访问频繁影响效率, 现在的操作系统已经这一块已经发生了变化。) chmod o-r proc.c 就是改变属性 vim proc.c 就是修改内容
但是内容被修改可能发生属性的联动, 因为文件的大小变了。
不管是源文件, 还是可执行程序, 都是文件, 程序可不可总是被执行就是看mtime的时间, 如果没有修改proc.c的m时间,只修改了proc.exe的时间, 那么表示exe比.c更新,编译器就认为没有必要再重新编译了, 相反如果.c的mtime时间更新,那么就可以重新编译生成新的可执行程序。
拓展, touch不仅可以创建文件, 还可以用来修改文件的时间(mtime和ctime都会被修改)。
结论: .PHONY:让make忽略源⽂件和可执⾏⽬标⽂件的M时间对⽐
执行make, 默认在命令行打印四条“hahaha”, 如果加上@则不会执行,关闭命令回显。
myproc:myproc.o
gcc myproc.o -o myproc
myproc.o:myproc.s
gcc -c myproc.s -o myproc.o
myproc.s:myproc.i
gcc -S myproc.i -o myproc.s
myproc.i:myproc.c
gcc -E myproc.c -o myproc.i
.PHONY:clean
clean:
rm -f *.i *.s *.o myproc
编译
$ make
gcc -E myproc.c -o myproc.i
gcc -S myproc.i -o myproc.s
gcc -c myproc.s -o myproc.o
gcc myproc.o -o myproc
make是如何⼯作的,在默认的⽅式下,也就是我们只输⼊make命令。那么:
% 是makefile中的通配符 例如: %.c: 当前目录下所有.c文件, 展开到依赖项中。 依赖关系:右侧的依赖文件, 一个一个的交给gcc -c选项, 形成同名的.o文件。
继续
@表示目标文件, ^表示所有依赖文件的文件列表
默认情况下, makefile执行到了一个目标文件之后就会停下来, 比如下图, 只会生成一个proc文件。
如果想进行多个文件同时编译,则可以添加目标all,但是不给执行方法, 为了形成目标all,makefile就会像下面搜索依赖项,形成proc和code。
小结一下:
BIN=proc.exe # 定义变量
CC=gcc
#SRC=$(shell ls *.c) # 采⽤shell命令⾏⽅式,获取当前所有.c⽂件名
SRC=$(wildcard *.c) # 或者使⽤ wildcard 函数,获取当前所有.c⽂件名
OBJ=$(SRC:.c=.o) # 将SRC的所有同名.c 替换 成为.o 形成⽬标⽂件列表
LFLAGS=-o # 链接选项
FLAGS=-c # 编译选项
RM=rm -f # 引⼊命令
$(BIN):$(OBJ)
@$(CC) $(LFLAGS) $@ $^ # $@:代表⽬标⽂件名。 $^: 代表依赖⽂件列表
@echo "linking ... $^ to $@"
%.o:%.c # %.c 展开当前⽬录下所有的.c。 %.o: 同时展开同
名.o
@$(CC) $(FLAGS) $< # %<: 对展开的依赖.c⽂件,⼀个⼀个的交给gcc。
@echo "compling ... $< to $@" # @:不回显命令
.PHONY:clean
clean:
$(RM) $(OBJ) $(BIN) # $(RM): 替换,⽤变量内容替换它
.PHONY:test
test:
@echo $(SRC)
@echo $(OBJ)
完