多個文件目錄下Makefile的寫法
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://blog.csdn.net/liangkaiming/article/details/6267357
Makefile中的wildcard用法
在Makefile規則中,通配符會被自動展開。但在變量的定義和函數引用時,通配符將失效。這種情況下如果需要通配符有效,就需要使用函數“wildcard”,它的用法是:$(wildcard PATTERN...) 。在Makefile中,它被展開為已經存在的、使用空格分開的、匹配此模式的所有文件列表。如果不存在任何符合此模式的文件,函數會忽略模式字符並返回空。需要註意的是:這種情況下規則中通配符的展開和上一小節匹配通配符的區別。
一般我們可以使用“$(wildcard *.c)”來獲取工作目錄下的所有的.c文件列表。復雜一些用法;可以使用“$(patsubst %.c,%.o,$(wildcard *.c))”,首先使用“wildcard”函數獲取工作目錄下的.c文件列表;之後將列表中所有文件名的後綴.c替換為.o。這樣我們就可以得到在當前目錄可生成的.o文件列表。因此在一個目錄下可以使用如下內容的Makefile來將工作目錄下的所有的.c文件進行編譯並最後連接成為一個可執行文件:
#sample Makefile
objects := $(patsubst %.c,%.o,$(wildcard *.c))
foo : $(objects)
cc -o foo $(objects)
這裏我們使用了make的隱含規則來編譯.c的源文件。對變量的賦值也用到了一個特殊的符號(:=)。
1、wildcard : 擴展通配符
2、notdir : 去除路徑
3、patsubst :替換通配符
例子:
建立一個測試目錄,在測試目錄下建立一個名為sub的子目錄
$ mkdir test
$ cd test
$ mkdir sub
在test下,建立a.c和b.c2個文件,在sub目錄下,建立sa.c和sb.c2 個文件
建立一個簡單的Makefile
src=$(wildcard *.c ./sub/*.c)
dir=$(notdir $(src))
obj=$(patsubst %.c,%.o,$(dir) )
all:
@echo $(src)
@echo $(dir)
@echo $(obj)
@echo "end"
執行結果分析:
第一行輸出:
a.c b.c ./sub/sa.c ./sub/sb.c
wildcard把 指定目錄 ./ 和 ./sub/ 下的所有後綴是c的文件全部展開。
第二行輸出:
a.c b.c sa.c sb.c
notdir把展開的文件去除掉路徑信息
第三行輸出:
a.o b.o sa.o sb.o
在$(patsubst %.c,%.o,$(dir) )中,patsubst把$(dir)中的變量符合後綴是.c的全部替換成.o,
任何輸出。
或者可以使用
obj=$(dir:%.c=%.o)
效果也是一樣的。
這裏用到makefile裏的替換引用規則,即用您指定的變量替換另一個變量。
它的標準格式是
$(var:a=b) 或 ${var:a=b}
它的含義是把變量var中的每一個值結尾用b替換掉a
今天在研究makefile時在網上看到一篇文章,介紹了使用函數wildcard得到指定目錄下所有的C語言源程序文件名的方法,這下好了,不用手工一個一個指定需要編譯的.c文件了,方法如下:
SRC = $(wildcard *.c)
等於指定編譯當前目錄下所有.c文件,如果還有子目錄,比如子目錄為inc,則再增加一個wildcard函數,象這樣:
SRC = $(wildcard *.c) $(wildcard inc/*.c)
也可以指定匯編源程序:
ASRC = $(wildcard *.S)
多個文件目錄下Makefile的寫法