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;