下面用两个例子写Makefile编译代码。
一个编译c语言代码,一个编译涉及到ffmpeg的代码
第一个例子:
可以看到我当前目录有这些C文件和一个Makefile,下面放出Makefile代码
详细注释写在了Makefile里面
#写一个通用的Makefile编译代码
#指定编译器
CC:=gcc
#CFLAGS 表示用于 C 编译器的选项,例如wall就是开启警告
#CXXFLAGS 表示用于 C++ 编译器的选项。
CFLAGS:=-Wall
#要生成的最终文件,举个例子,正常执行gcc编译c文件会产生a.out,a.out就是最终的文件
EXE:=test.out test1.out test2.out test3.out
#这个就是保护中间文件。如果不加的话,中间生成的.o文件会被删除
.SECONDARY:
#编译的总入口,按照顺序编译,先编译目录dir 第二个是$(addprefix bin/, $(EXE)) 最后是tag
#$(addprefix bin/, $(EXE))的意思是bin下的EXE文件,展开就是bin/test.out .....bin/test3.out这4个文件
all:dir $(addprefix bin/, $(EXE)) tag
dir:
mkdir -p bin
mkdir -p obj
tag:
ctags *.c
# $< 第一个依赖文件(即冒号右边的文件,以下面的例子来说,就是指obj/%.o)
# $^ 所有依赖文件
# $@ 编译的目标文件(即是要生成的目标,冒号左边的文件,以下面的例子来说,就是指obj/%.out)
# %是通配符
# $(XXX)就是取值展开
bin/%.out:obj/%.o
$(CC) $(CFLASG) $< -o $@
obj/%.o:%.c
$(CC) $(CFLASG) $< -c -o $@
#执行make clean就会执行下面三条命令,一般用来清除编译环境
clean:
rm bin -rf
rm obj -rf
rm tags
注意,Makefile的编译规则是逆序去寻找的,例如要编译生成test.out,那么先去找生成test.out的依赖文件,发现test.out是由test.o编译生成的。
接着发现test.o文件不存在,那么会继续去找test.o的依赖文件,然后发现是test.c文件。
至此全部找到之后就会编译test.c生成test.o,再编译test.o生成最终我们要的test.out可执行文件
最终make的效果如下图:
第二个例子:ffmpeg的代码编译使用Makefile来管理
编译ffmpeg的代码有时要加上很多库和库的路径,每次编译加上一大堆东西显得格外麻烦,用Makefile编译就可以省去很多功夫。
CC:=gcc
INCLUDES:=$(shell pkg-config --cflags libavformat libavcodec libswresample libswscale libavutil sdl)
CFLAGS:=-Wall -ggdb
LDFLAGS:=$(shell pkg-config --libs libavformat libavcodec libswresample libswscale libavutil sdl) -lm
EXE:=tutorial01.out tutorial02.out tutorial03.out tutorial04.out
.SECONDARY:
all: dirs $(addprefix bin/, $(EXE)) tags
dirs:
mkdir -p obj
mkdir -p bin
tags: *.c
ctags *.c
bin/%.out: obj/%.o
$(CC) $(CFLAGS) $< $(LDFLAGS) -o $@
obj/%.o : %.c
$(CC) $(CFLAGS) $< $(INCLUDES) -c -o $@
clean:
rm -f obj/*
rm -f bin/*
rm -f tags
可以看到下图的结果。
Makefile:19: *** missing separator. Stop.
遇到这个问题的话,是因为句首不是TAB,句首改为TAB就可以了