1. 程式人生 > >Makefile 詳解一(含通用模板)

Makefile 詳解一(含通用模板)

介紹

Makefile是一個規定了怎麼去編譯和連結程式的指令碼檔案,在執行make命令時會執行該檔案,window環境下的IDE,如visual studio
已經集成了該功能,不需要關心程式的編譯規則,在linux下做C/C++開發時經常用到,說到這裡首先要知道一個工具make。

make是一個解釋Makefile中指令的命令工具,常見的IDE都集成了這個工具。目前centos 7.3 GNU的make版本是3.82

規則

目標檔案:依賴檔案
[Tab]系統指令1     (注意:系統指令前必須有tab)

使用

3.1 簡單使用

現有檔案main.cpp  test.cpp test.h三個檔案,說些Makefile實現增量編譯(當其中有一個檔案變化時,重新編譯該檔案)
helloworld: main.o test.o
    g++ main.o test.o -o helloworld

main.o: main.cpp test.h
    g++ -c main.cpp -o main.o

test.o: test.cpp test.h
    g++ -c test.cpp -o test.o

clean:
    rm *.o helloworld

3.2 makefile中使用註釋,變數和函式

註釋:
    行前面加"#"號,如#g++ main.o test.o -o helloworld  表示註釋了該行
變數:
    用=定義一個變數,並且賦值(等號兩邊可以加空格)
    用+=追加字串
    用$(A)來取得變數的值

例:    
A = src
    echo $(A)  
    @echo $(A)  ##只輸出echo的結果,不顯示執行的命令

特殊變數:
    $@ 目標檔案
    $^ 依賴項列表
    $< 依賴項列表第一項


函式:
    Makefile中有一些預定義函式,形式:
    $(函式名 引數列表)
    引數列表:以逗號分隔
    函式名和引數之間用空格分開

    #獲取當前目錄路徑
    PWD = $(shell pwd)

    #獲取當前目錄下所有.cpp檔案
    CXX_SOURCES = $(wildcard *.cpp)

    #獲取當前目錄下所有.cpp檔案編譯後的所有目標檔案.o
    CXX_OBJECTS = $(patsubst *.cpp *.o, $(CXX_SOURCES))

3.3 優化3.1中Makefile

EXE = helloworld
GCC = g++

$(EXE): main.o test.o
        $(GCC) $^ -o $(EXE)


main.o: main.cpp test.h
        $(GCC) -c $< -o $@


test.o: test.cpp test.h
        $(GCC) -c $< -o $@

clean:
        rm *.o $(EXE)

3.4 優化3.3中Makefile

繼續優化Makefile,新增資料夾,把原始碼都放入src和lib資料夾,保持增量編譯,即為Makefile通用模板
EXE = helloworld
GCC = g++
SUBDIR = src lib

CPP_SOURCES = $(foreach dir, $(SUBDIR), $(wildcard $(dir)/*.cpp))
CPP_OBJECTS = $(patsubst %.cpp, %.o, $(CPP_SOURCES))
DEP_FILES = $(patsubst %.o, %.d, $(CPP_OBJECTS))

$(EXE): $(CPP_OBJECTS)
        $(GCC) $(CPP_OBJECTS) -o $@

%.o: %.cpp
        $(GCC) -c -MMD $< -o $@

-include $(DEP_FILES)

clean:
        rm  $(CPP_OBJECTS)  $(EXE)


此Makefile可以作為通用Makefile模板來編譯C/C++專案,歡迎收藏