您的位置:首页 > 其它

多个文件目录下Makefile的写法

2014-01-04 19:38 471 查看
1、前言
  目前从事于linux下程序开发,涉及到多个文件,多个目录,这时候编译文件的任务量比较大,需要写Makefile。关于Makefile的详细内容可以参考网上流传非常广泛的《跟我一起写Makefile》http://blog.csdn.net/haoel/article/details/2886/,作者是个大牛,非常佩服。
2、简单测试
  测试程序在同一个文件中,共有func.h、func.c、main.c三个文件,Makefile写法如下所示:

1 CC = gcc
2 CFLAGS = -g -Wall
3
4 main:main.o func.o
5     $(CC)  main.o func.o -o main
6 main.o:main.c
7     $(CC) $(CFLAGS)  -c main.c -o main.o
8 func.o:func.c
9     $(CC) $(CFLAGS) -c func.c -o func.o
10 clean:
11     rm -rf *.o


执行过程如下图所示:



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


完整的Makefile如下所示:

1 DIR_INC = ./include
2 DIR_SRC = ./src
3 DIR_OBJ = ./obj
4 DIR_BIN = ./bin
5
6 SRC = $(wildcard ${DIR_SRC}/*.c)
7 OBJ = $(patsubst %.c,${DIR_OBJ}/%.o,$(notdir ${SRC}))
8
9 TARGET = main
10
11 BIN_TARGET = ${DIR_BIN}/${TARGET}
12
13 CC = gcc
14 CFLAGS = -g -Wall -I${DIR_INC}
15
16 ${BIN_TARGET}:${OBJ}
17     $(CC) $(OBJ)  -o $@
18
19 ${DIR_OBJ}/%.o:${DIR_SRC}/%.c
20     $(CC) $(CFLAGS) -c  $< -o $@
21 .PHONY:clean
22 clean:
23     find ${DIR_OBJ} -name *.o -exec rm -rf {}


解释如下:
(1)Makefile中的 符号 $@, $^, $< 的意思:

  $@ 表示目标文件

  $^ 表示所有的依赖文件

  $< 表示第一个依赖文件

  $? 表示比目标还要新的依赖文件列表
(2)wildcard、notdir、patsubst的意思:
  wildcard : 扩展通配符

  notdir : 去除路径

  patsubst :替换通配符
例如下图例子所示:



输出结果如下所示:



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.groad.net/bbs/read.php?tid-2920-page-e.html
/article/7822320.html
/article/7999887.html


一、例子

(1)makefile和src源文件不在同一目录下

(2)把.o生成到指定目录下

文件结构目录

  ----inc //放头文件

  ----lib   //放所需要的.a或者.so文件

  ----src  //放置源文件

  ----obj  //放生成的.o文件

  ----Makefile //makefile文件

inc有头文件Hi_String.h

src有源文件Hi_String.cpp

#compile and lib parameter
#编译参数
#-g表示可以使用gdb调试
#-Wall是所有错误和警告在编译的时候都打出来
#-O2: 包含-O1的优化并增加了不需要在目标文件大小和执行速度上进行折衷的优化.
#      编译器不执行循环展开以及函数内联.此选项将增加编译时间和目标文件的执行性能.
CC      := g++
CCOPT := -g -Wall -O2

#当前makefile路径
CURRENT_PATH := $(shell pwd)
#头文件路径
INC_PATH := $(CURRENT_PATH)/inc
#lib库路径
LIB_PATH := $(CURRENT_PATH)/lib
#源文件路径
SRC_PATH := $(CURRENT_PATH)/src
#object文件路径
OBJS_PATH := $(CURRENT_PATH)/obj

#源文件,自动找所有.cpp文件,并将目标定义为同名.o文件
#wildcard:扩展通配符 notdir:去除路径 patsubst:替换通配符
SRC_FILE_DIR := $(wildcard $(SRC_PATH)/*.cpp)
SRC_FILE :=$(notdir $(SRC_FILE_DIR))
#把所有.cpp变成.o文件;这里有两种表达方法
#(1)OBJS_FILE := $(SRC_FILE:.cpp=.o)
#(2)OBJS_FILE := $(patsubst %.cpp,%.o,$(SRC_FILE))
OBJS_FILE := $(SRC_FILE:.cpp=.o)

OBJS:=$(OBJS_PATH)/$(OBJS_FILE)

TARGET := Hi_String

$(OBJS_PATH)/%.o:$(SRC_PATH)/%.cpp
$(CC) $(CCOPT) -I$(INC_PATH) -c $^ -o $@

$(TARGET) : $(OBJS)
$(CC) $(CCOPT) -I$(INC_PATH) -o $@ $^

.PHONY : clean
clean :
rm -rf $(TARGET)
rm -rf $(OBJS)


  这样会把.o文件生成到obj里面;

   把最终可执行文件生成到Makefile一起的目录下
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: