分享一个比较通用的Makefile

来源:互联网 发布:中国地图编辑软件 编辑:程序博客网 时间:2024/06/10 08:54

最近学习了解了Makefile, 这里简单记录下以备忘. 先贴出一个简单的Makefile .

#这里指预处理的头文件目录在当前的include目录下CPPFLAGS= -Iinclude#表示 编译时2级优化,包含调试信息,供gdb调试CFLAGS= -O2 -g#共享库,没有可以不写LDFLAGS=#用CC变量表示gcc,或g++等CC=gccsrc=$(wildcard *.c) $(wildcard src/*.c)obj=$(patsubst %.c,%.o, $(src))target=app$(target):$(obj)    $(CC) $^ $(LDFLAGS) -o $@%.o:%.c    $(CC) -c $^ $(CPPFLAGS) $(CFLAGS) -o $@#声明一个伪目标,因为如果有同名的clean文件,因为clean没有依赖,所以将不会执行这个目标,#这里-rm中的-表示如果此命令出错,make clean 会继续执行后面的命令,而不是退出Makefile.PHONY: cleanclean:    -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 :
    @echo $(src) //这里的@表示不显示命令本身,只显示结果
    @echo $(obj)
    可以用make test 查看

  • @^表示所有依赖,$<表示依赖中的第一个, 这里Makefile的三要素就是目标,依赖,命令, 写Makefile时,要从上向下建立关系树,从下向上执行,所以
    可执行文件(a.out) ->目标文件(.o) ->源文件(.c/.cpp).
    • 执行时先执行
      %.o:%.c
      $(CC) -c $^ $(CPPFLAGS) $(CFLAGS) -o $@
      等价于gcc -c $^ -Iinclude -O2 -g -o $@
      这里只编译不链接
      $@表示目标,也就是对应的.o文件 ,$^表示所有依赖,也就是所有的.c文件
    • 再执行
      $(target):$(obj)
      $(CC) $^ $(LDFLAGS) -o $@
      表示链接生成的所有的.o文件, 同时链接时才加载共享库,这里target变量为指定生成的可执行文件的名字为app, 自己按需要指定.

2. 这种情况下的小项目,include目录中为头文件.h, src目录中为源文件.c , lib为库文件.so, 可以简单的上面的Makefile实现,直接make.

这里写图片描述

3 .如果要是多个文件目录的项目

这里写图片描述

当程序文件比较大,这时候对文件进行分类,分为头文件(include)、源文件(src)、目标文件(obj)、可执行文件(bin), 库文件(lib)。将文件按照文件类型放在不同的目录当中,这个时候的Makefile需要统一管理这些文件,将生产的目标文件放在目标目录下,可执行文件放在可执行文件目录下:

DIR_INC = ./includeDIR_SRC = ./srcDIR_OBJ = ./objDIR_BIN = ./bin#这里指预处理的头文件目录在当前的include目录下CPPFLAGS= -I${DIR_INC}#表示 编译时2级优化,包含调试信息,供gdb调试CFLAGS= -O2 -g#共享库,没有可以不写LDFLAGS=#用CC变量表示gcc,或g++等CC=gccSRC = $(wildcard ${DIR_SRC}/*.c)  OBJ = $(patsubst %.c,${DIR_OBJ}/%.o,$(notdir ${SRC})) TARGET = appBIN_TARGET = ${DIR_BIN}/${TARGET}${BIN_TARGET}:${OBJ}    $(CC) $(OBJ) $(LDFLAGS) -o $@${DIR_OBJ}/%.o:${DIR_SRC}/%.c    $(CC) -c $^ $(CPPFLAGS) $(CFLAGS) -o $@.PHONY:cleanclean:    find ${DIR_OBJ} -name *.o -exec rm -rf {}

参考:

https://www.cnblogs.com/Anker/p/3242207.html