多个文件目录下Makefile的写法

本文介绍了一种在多个文件目录下编写Makefile的方法,适用于Linux下的程序开发。文章通过一个包含func.h、func.c和main.c的小型示例项目,展示了如何组织Makefile以简化编译流程,并提供了一个更复杂的多目录项目的Makefile模板。

多个文件目录下Makefile的写法

1、前言

  目前从事于linux下程序开发,涉及到多个文件,多个目录,这时候编译文件的任务量比较大,需要写Makefile。关于Makefile的详细内容可以参考网上流传非常广泛的《跟我一起写Makefile》http://blog.youkuaiyun.com/haoel/article/details/2886/,作者是个大牛,非常佩服。

2、简单测试

  测试程序在同一个文件中,共有func.h、func.c、main.c三个文件,Makefile写法如下所示:

  1. CC=gcc  
  2. objects=obj/main.o obj/printStatus.o  
  3.   
  4. bin/main:$(objects)  
  5.    $(CC) -o bin/main $(objects)  
  6.   
  7. obj/main.o:src/main.c include/common.h  
  8.    $(CC) -o obj/main.o -c src/main.c -Iinclude  
  9.   
  10. obj/printStatus.o:src/printStatus.c include/common.h  
  11.    $(CC) -o obj/printStatus.o -c src/printStatus.c -Iinclude  
  12.   
  13. clean:  
  14.    rm -rf $(objects) bin/main 

执行过程如下图所示:

3、通用模板

  实际当中程序文件比较大,这时候对文件进行分类,分为头文件、源文件、目标文件、可执行文件。也就是说通常将文件按照文件类型放在不同的目录当中,这个时候的Makefile需要统一管理这些文件,将生产的目标文件放在目标目录下,可执行文件放到可执行目录下。测试程序如下图所示:

完整的Makefile如下所示:


  1. #把所有的目录做成变量,方便修改和移植   
  2. BIN = ./bin  
  3.  SRC = ./src  
  4.  INC = ./include  
  5.  OBJ = ./obj  
  6.    
  7. #提前所有源文件(即:*.c文件)和所有中间文件(即:*.o)  
  8.  SOURCE = $(wildcard ${SRC}/*.c)  
  9.  OBJECT = $(patsubst %.c,${OBJ}/%.o,$(notdir ${SOURCE}))  
  10.    
  11. #设置最后目标文件  
  12.  TARGET = main  
  13.  BIN_TARGET = ${BIN}/${TARGET}  
  14.    
  15.  CC = gcc   
  16.  CFLAGS = -g -Wall -I${INC}   
  17.    
  18. #用所有中间文件生成目的文件,规则中可以用 $^替换掉 ${OBJECT}  
  19.  ${BIN_TARGET}:${OBJECT}  
  20.      $(CC) -o $@ ${OBJECT}  
  21.    
  22. #生成各个中间文件  
  23.  ${OBJ}/%.o:${SRC}/%.c   
  24.      $(CC) $(CFLAGS) -o $@ -c $<  
  25.    
  26.  .PHONY:clean  
  27.  clean:  
  28.      find $(OBJ) -name *.o -exec rm -rf {} \; #这个是find命令,不懂的可以查下资料  
  29.      rm -rf $(BIN_TARGET) 
在看多目录的makefile时,先来理解下几个函数和变量;

(1)Makefile中的 符号 $@, $^, $< 的意思:
  $@  表示目标文件
  $^  表示所有的依赖文件
  $<  表示第一个依赖文件
  $?  表示比目标还要新的依赖文件列表

(2)wildcard、notdir、patsubst的意思:

  wildcard : 扩展通配符
  notdir : 去除路径
  patsubst :替换通配符

解释如下:

函数:

        wildcard 这是扩展通配符函数,功能是展开成一列所有符合由其参数描述的文 件名,文件间以空格间隔;比如:罗列出src下的所有.c文件:$(wildcard ${SRC}/*.c)

        patsubst 这是匹配替换函数, patsubst ( 需要匹配的文件样式,匹配替换成什么文件,需要匹配的源文件)函数。比如:用src下的*.c替换成对应的 *.o文件存放到obj中:$(patsubst  %.c, ${OBJ}/%.o, $(notdir $(SOURCE)))

        notdir 这是去除路径函数,在上面patsubst函数中已经使用过,去除SOURCE中文件的所有目录,只留下文件名;


变量:

        $@:表示目标文件;一般是在规则中这么用:gcc  -o $@  $(object);

        $^:表示所有依赖文件;一般是在规则中这么用:gcc -o $@  $^  ;用所有依赖文件链接成目的文件;

        $<:表示第一个依赖文件;在规则中使用:gcc -o $@ -c $< ;其实这个时候就是每个依赖文件生成一个目的文件;


 例如下图例子所示:

输出结果如下所示:

SRC = $(wildcard *.c)

等于指定编译当前目录下所有.c文件,如果还有子目录,比如子目录为inc,则再增加一个wildcard函数,象这样:

SRC = $(wildcard *.c) $(wildcard inc/*.c)

(3)gcc -I -L -l的区别:

       gcc -o hello hello.c -I /home/hello/include -L /home/hello/lib -lworld

       上面这句表示在编译hello.c时-I /home/hello/include表示将/home/hello/include目录作为第一个寻找头文件的目录,

   寻找的顺序是:/home/hello/include-->/usr/include-->/usr/local/include

   -L /home/hello/lib表示将/home/hello/lib目录作为第一个寻找库文件的目录,

   寻找的顺序是:/home/hello/lib-->/lib-->/usr/lib-->/usr/local/lib

       -lworld表示在上面的lib的路径中寻找libworld.so动态库文件(如果gcc编译选项中加入了“-static”表示寻找libworld.a静态库文件)


转载:http://www.cnblogs.com/Anker/p/3242207.html

<think>好的,用户想了解如何在一个多目录项目中使用Makefile。首先,我需要确定他们的使用场景。多目录项目通常意味着源代码分布在不同的子目录中,比如src、lib、include等,而构建输出可能放在bin或build目录中。用户可能希望用Makefile来自动化编译和链接过程,这样就不需要每次都手动输入复杂的编译命令。 接下来,用户可能对Makefile的基本结构不太熟悉,需要从基础讲起。比如,目标、依赖、命令这些概念。然后,如何处理多个目录中的源文件?可能需要设置变量来指定各个目录的路径,例如SRC_DIR、OBJ_DIR、BIN_DIR等。这样Makefile可以自动在这些目录中查找源文件,并将生成的目标文件放到对应的位置。 然后,需要考虑如何自动生成依赖关系,特别是当源文件分散在不同目录时。可能需要使用通配符来收集所有的.c文件,然后替换成.o文件,并指定输出到OBJ_DIR。同时,需要处理头文件的包含,确保编译器能找到头文件的位置,比如使用-I选项指定include目录。 另一个重要的点是自动化依赖生成。可以使用gcc的-M选项来生成每个源文件的依赖关系,这样当头文件更改时,Makefile能够知道需要重新编译哪些源文件。这部分可能有点复杂,用户可能需要一个简单的例子来理解如何将生成的依赖文件包含到Makefile中。 还要考虑目标文件的存放位置,确保OBJ_DIR存在,可以使用mkdir -p命令在编译前创建目录。另外,清理操作也很重要,用户可能希望make clean能删除所有生成的文件,包括不同目录中的目标文件和可执行文件。 最后,用户可能需要一个完整的示例,展示如何将所有部分整合在一起。这个示例应该包括目录结构的假设,比如src存放源文件,include存放头文件,bin存放可执行文件,obj存放目标文件。然后逐步解释Makefile中的每个部分,包括变量定义、目标规则、自动生成依赖等。 需要注意的是,用户可能对Makefile的高级特性不熟悉,所以需要保持示例简单明了,避免复杂的逻辑。同时,要提醒他们根据实际项目结构调整目录路径和变量名。可能还需要指出常见的陷阱,比如路径中的空格问题,或者不同操作系统下命令的差异(虽然Makefile通常是跨平台的,但某些命令如rm可能需要调整)。 总结来说,用户需要一个结构清晰、能够处理多目录项目的Makefile示例,以及对其各部分功能的解释,帮助他们理解如何组织和管理稍复杂的项目结构。同时,要确保解释的语言简洁,避免技术术语过多,让初学者也能理解。</think>当项目涉及多个目录时,编写 Makefile 需要处理不同目录下的源代码、头文件和生成文件的路径管理。以下是一个典型的多目录结构示例及对应的 Makefile 写法: --- ### 📂 项目结构 ``` project/ ├── src/ # 存放源代码 *.c ├── include/ # 存放头文件 *.h ├── obj/ # 存放编译生成的 *.o 文件 └── bin/ # 存放最终可执行文件 ``` --- ### 📜 Makefile 示例 ```makefile # 编译器配置 CC = gcc CFLAGS = -I./include -Wall -g # 目录定义 SRC_DIR = src OBJ_DIR = obj BIN_DIR = bin # 自动获取所有 .c 文件并生成对应的 .o 文件路径 SOURCES = $(wildcard $(SRC_DIR)/*.c) OBJECTS = $(patsubst $(SRC_DIR)/%.c, $(OBJ_DIR)/%.o, $(SOURCES)) TARGET = $(BIN_DIR)/myapp # 默认目标 all: $(TARGET) # 链接目标文件生成可执行文件 $(TARGET): $(OBJECTS) | $(BIN_DIR) $@ # 编译每个 .c 文件到 .o 文件(自动处理头文件依赖) $(OBJ_DIR)/%.o: $(SRC_DIR)/%.c | $(OBJ_DIR) $(CC) $(CFLAGS) -c $< -o $@ # 自动创建目录 $(BIN_DIR) $(OBJ_DIR): mkdir -p $@ # 清理生成文件 clean: rm -rf $(OBJ_DIR) $(BIN_DIR) # 声明伪目标 .PHONY: all clean ``` --- ### 🔑 关键点解析 1. **路径管理** - 使用 `SRC_DIR`, `OBJ_DIR`, `BIN_DIR` 变量定义目录路径 - `wildcard` 自动收集源文件,`patsubst` 转换路径到目标文件 2. **目录自动创建** - `| $(BIN_DIR)` 表示目录是顺序依赖(Order-only prerequisites) - `mkdir -p` 确保编译前目录存在 3. **头文件依赖处理** - `-I./include` 指定头文件搜索路径 - 进阶用法:可通过 `-MMD` 自动生成 `.d` 依赖文件(需配合 `include` 指令) 4. **可扩展性** - 添加新文件时无需修改 Makefile - 支持多层级目录(需调整 `wildcard` 模式) --- ### 🚀 使用方式 ```bash # 编译项目 make # 清理生成文件 make clean ``` --- 如果需要更复杂的依赖管理(如自动检测头文件修改),可以通过 `gcc -MMD` 生成依赖文件,但这会增加 Makefile 的复杂度。此示例是一个平衡简洁性与实用性的基础模板。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值