1. 程式人生 > >大型工程多個目錄下的Makefile寫法

大型工程多個目錄下的Makefile寫法

qt5 pan ron 指定 com exec bsp 可執行文件 不同

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

http://blog.csdn.net/liangkaiming/article/details/6267357

http://blog.csdn.net/zqt520/article/details/7727051

大型工程多個目錄下的Makefile寫法