分享一个比较通用的Makefile
程序员文章站
2022-07-02 17:03:01
...
最近学习了解了Makefile, 这里简单记录下以备忘. 先贴出一个简单的Makefile .
#这里指预处理的头文件目录在当前的include目录下
CPPFLAGS= -Iinclude
#表示 编译时2级优化,包含调试信息,供gdb调试
CFLAGS= -O2 -g
#共享库,没有可以不写
LDFLAGS=
#用CC变量表示gcc,或g++等
CC=gcc
src=$(wildcard *.c) $(wildcard src/*.c)
obj=$(patsubst %.c,%.o, $(src))
target=app
$(target):$(obj)
$(CC) $^ $(LDFLAGS) -o [email protected]
%.o:%.c
$(CC) -c $^ $(CPPFLAGS) $(CFLAGS) -o [email protected]
#声明一个伪目标,因为如果有同名的clean文件,因为clean没有依赖,所以将不会执行这个目标,
#这里-rm中的-表示如果此命令出错,make clean 会继续执行后面的命令,而不是退出Makefile
.PHONY: clean
clean:
-rm -f *.o
-rm -f $(target)
1 . 在Makefile中, #表示注释行. 这里的
CFLAGS 预处理器标志,也就是预处理时的参数-I,-E。(如头文件的位置)
LDFLAGS 加载时的参数(如共享库)
wildcard 表示查找符合条件的文件名 ,
$(wildcard *.c) 表示在当前目录下查找 .c 文件,
$(wildcard src/*.c) 表示在当前目录下的src目录下查找 .c 文件src=$(wildcard *.c) $(wildcard src/*.c)
表示把在当前目录下的查找的.c
文件和在src目录下查找的.c文件赋给变量src .obj=$(patsubst %.c,%.o, $(src))
表示模式查找替换,也就是将src中的所有.c文件替换为.o文件,但是注意这里并没有生成.o文件.
可以用:test :
可以用make test 查看
@echo $(src) //这里的@表示不显示命令本身,只显示结果
@echo $(obj)-
[email protected]表示目标 , $^表示所有依赖,$<表示依赖中的第一个
, 这里Makefile的三要素就是目标,依赖,命令, 写Makefile时,要从上向下建立关系树,从下向上执行,所以
可执行文件(a.out) ->目标文件(.o) ->源文件(.c/.cpp). - 执行时先执行
%.o:%.c
等价于
$(CC) -c $^ $(CPPFLAGS) $(CFLAGS) -o [email protected]gcc -c $^ -Iinclude -O2 -g -o [email protected]
这里只编译不链接
[email protected]表示目标,也就是对应的.o文件 ,$^表示所有依赖,也就是所有的.c文件 - 再执行
$(target):$(obj)
$(CC) $^ $(LDFLAGS) -o [email protected]
表示链接生成的所有的.o文件, 同时链接时才加载共享库,这里target变量为指定生成的可执行文件的名字为app, 自己按需要指定.
2. 这种情况下的小项目,include目录中为头文件.h, src目录中为源文件.c , lib为库文件.so, 可以简单的上面的Makefile实现,直接make.
3 .如果要是多个文件目录的项目
当程序文件比较大,这时候对文件进行分类,分为头文件(include)、源文件(src)、目标文件(obj)、可执行文件(bin), 库文件(lib)。将文件按照文件类型放在不同的目录当中,这个时候的Makefile需要统一管理这些文件,将生产的目标文件放在目标目录下,可执行文件放在可执行文件目录下:
DIR_INC = ./include
DIR_SRC = ./src
DIR_OBJ = ./obj
DIR_BIN = ./bin
#这里指预处理的头文件目录在当前的include目录下
CPPFLAGS= -I${DIR_INC}
#表示 编译时2级优化,包含调试信息,供gdb调试
CFLAGS= -O2 -g
#共享库,没有可以不写
LDFLAGS=
#用CC变量表示gcc,或g++等
CC=gcc
SRC = $(wildcard ${DIR_SRC}/*.c)
OBJ = $(patsubst %.c,${DIR_OBJ}/%.o,$(notdir ${SRC}))
TARGET = app
BIN_TARGET = ${DIR_BIN}/${TARGET}
${BIN_TARGET}:${OBJ}
$(CC) $(OBJ) $(LDFLAGS) -o [email protected]
${DIR_OBJ}/%.o:${DIR_SRC}/%.c
$(CC) -c $^ $(CPPFLAGS) $(CFLAGS) -o [email protected]
.PHONY:clean
clean:
find ${DIR_OBJ} -name *.o -exec rm -rf {}
参考: