makefile學習經驗(二)----編譯生成靜態庫文件


Lunix下編譯靜態庫文件:

.o后綴文件:編譯生成的中間代碼文件;

.a后綴文件:靜態庫文件,編譯的時候會合到可執行程序中,文件比較大;

.so后綴文件:動態庫文件,只是一個指向,不會合到可執行程序中,當要調用函數庫的時候才使用;

 

用一個簡單的例子闡述下在lunix環境下生成靜態庫文件的過程:

一.   源碼文件:

so_test.h:

void test_a();

void test_b();

void test_c();

 

test_a.c:

#include "so_test.h"

void test_a()

{

    printf("this is in test_a...\n");

}

 

test_b.c:

#include "so_test.h"

void test_a()

{

    printf("this is in test_b...\n");

}

 

test_c.c:

#include "so_test.h"

void test_a()

{

    printf("this is in test_c...\n");

}

 

test.c:

#include "so_test.h"

int main()

{

test_a();

test_b();

test_c();

return 0;

}

 

二.   makefile 文件:

makefile:

#*****************************************************************************

# Copyright        :  

#

# Author           :   kimi

# Date             :   2012-08-17

# Version          :   Demo2 靜態鏈接庫

# Description  :   Demo

#

#****************************************************************************/

 

SHELL = /bin/sh

 

LIB_DIR = /cbs/lhbb/hcj/dy/lib/

BIN_DIR = /cbs/lhbb/hcj/dy/bin/

OBJECT_DIR = /cbs/lhbb/dy/test/obj/

APP_DIR = /cbs/lhbb/hcj/dy/testapp/

 

$(shell mkdir -p ${LIB_DIR})

$(shell mkdir -p ${BIN_DIR})

$(shell mkdir -p ${OBJECT_DIR})

 

RM = rm -fr

#****************************************************************************

 

CC = gcc

AR = ar rc

 

SRC_OBJECT = $(APP_DIR)test_a.c $(APP_DIR)test_b.c $(APP_DIR)test_c.c

H_OBJECT = $(APP_DIR)so_test.h

OBJECT = test_a.o test_b.o test_c.o

 

DY_SRC_OBJECT = $(APP_DIR)test.c

DY_OBJECT=test.o

 

LIB_OBJECT = libtest.a

BIN_OBJECT = test

 

#****************************************************************************

 

.PHONY:all

all:$(LIB_OBJECT) $(BIN_OBJECT)

 

$(LIB_OBJECT):$(OBJECT)

        $(AR) $(LIB_OBJECT) $(OBJECT)

        mv $(LIB_OBJECT) $(LIB_DIR)

 

$(OBJECT):$(SRC_OBJECT) $(H_OBJECT)

        $(CC) -c $(SRC_OBJECT)

 

$(BIN_OBJECT):$(DY_OBJECT)

        $(CC) -o $(BIN_OBJECT) $(OBJECT_DIR)$(DY_OBJECT) $(LIB_DIR)$(LIB_OBJECT)

        mv $(BIN_OBJECT) $(BIN_DIR)

 

$(DY_OBJECT):$(DY_SRC_OBJECT)

        $(CC) -c $(DY_SRC_OBJECT) -o $(DY_OBJECT)

 

        mv $(OBJECT) $(DY_OBJECT) $(OBJECT_DIR)

 

clean:

        $(RM) $(LIB_DIR) $(BIN_DIR) $(OBJECT_DIR)

 

 

三.makefile文件的簡單闡述:

1. 路徑變量賦值:

LIB_DIR = /cbs/lhbb/hcj/dy/lib/          -----存放生成的靜態庫文件,.a文件;

BIN_DIR = /cbs/lhbb/hcj/dy/bin/         -----存放生成的可執行文件;

OBJECT_DIR = /cbs/lhbb/dy/test/obj/     -----存放生成的中間代碼文件,.o文件;

APP_DIR = /cbs/lhbb/hcj/dy/testapp/     -----存放源文件;

 

2.調用shell腳本,創建目錄:

$(shell mkdir -p ${LIB_DIR})

$(shell mkdir -p ${BIN_DIR})

$(shell mkdir -p ${OBJECT_DIR})

 

3.編譯變量賦值:

RM = rm –fr

CC = gcc

AR = ar rc       -----生成靜態庫文件命令

 

4.文件變量賦值:

SRC_OBJECT = $(APP_DIR)test_a.c $(APP_DIR)test_b.c $(APP_DIR)test_c.c

----源文件變量;

H_OBJECT = $(APP_DIR)so_test.h

----頭文件變量;

OBJECT = test_a.o test_b.o test_c.o

----中間代碼文件變量;

DY_SRC_OBJECT = $(APP_DIR)test.c

----生成test可執行文件所依賴的源文件變量;

DY_OBJECT=test.o

----編譯test.c生成的中間代碼文件變量;

LIB_OBJECT = libtest.a

----靜態庫文件變量;

BIN_OBJECT = test

----生成的可執行文件變量;

 

這個makefile中基本都是使用的變量,這樣編譯不同的工程的時候,就只要修改變量的值,編譯的整體框架就不需要修改,當然,這個框架只是本人剛開始研究makefile寫的,不夠好,望大神吐槽,謝謝!

 

5.編譯主體闡述:

.PHONY:all

----固定用.PHONY定義一個偽目標,偽目標不是文件,只是一個標簽,沒有依賴文件;這邊定義了一個偽目標all;

 

all:$(LIB_OBJECT) $(BIN_OBJECT)

----目標all包含了兩個需要生成的目標文件,一個是需要生成的靜態庫文件libtest.a,一個是最終生成的可執行文件test;但是為什么這邊需要用一個all來包含兩個目標文件呢?原因是通常情況下,系統只會去編譯生成makefile文件開頭最早一個目標文件,下面的目標文件是不會去執行編譯的,所以我在這邊將兩個目標文件寫在了一個偽目標all中,這樣就可以實現生成兩個目標文件;

 

生成靜態庫文件的過程:

$(LIB_OBJECT):$(OBJECT)      -----目標文件:依賴文件    libtest.a:test_a.o test_b.o test_c.o

        $(AR) $(LIB_OBJECT) $(OBJECT)    ----編譯生成靜態庫文件命令,特點就是需要使用

                        ar rc命令去編譯;

        mv $(LIB_OBJECT) $(LIB_DIR)      ----將生成的文件移動到對應的文件夾;

 

$(OBJECT):$(SRC_OBJECT) $(H_OBJECT)

        $(CC) -c $(SRC_OBJECT)

 

生成可執行文件test的過程:

$(BIN_OBJECT):$(DY_OBJECT)

        $(CC) -o $(BIN_OBJECT) $(OBJECT_DIR)$(DY_OBJECT) $(LIB_DIR)$(LIB_OBJECT)

        -----將靜態庫文件包含在命令中;

        mv $(BIN_OBJECT) $(BIN_DIR)

 

$(DY_OBJECT):$(DY_SRC_OBJECT)

        $(CC) -c $(DY_SRC_OBJECT) -o $(DY_OBJECT)

        mv $(OBJECT) $(DY_OBJECT) $(OBJECT_DIR)

 

執行make clean刪除生成的文件:

clean:

        $(RM) $(LIB_DIR) $(BIN_DIR) $(OBJECT_DIR)

 

四.注意點:

1.命令行必須以tab鍵打頭,否則會報錯;

2.生成靜態庫文件用命令:ar rc;

3.庫文件以lib打頭,.a為后綴;

 

五.執行結果:

 

 

bin文件夾中存放可執行文件test,運行./test結果如下:

this is in test_a...

this is in test_b...

this is in test_c...

 

lib文件夾下存放生成的靜態庫文件libtest.a;


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM