linux靜態庫及動態庫的建立
2007/01/03 11:38
瀏覽974
迴響1
推薦0
引用0
UNIX系統及各種軟件包為開發人員提供了大量的庫文件。但一般情況下這些庫文件還不能足以滿足用戶的所有需求。開發人員大多會根據他們自己的開發、研究要求編寫出許多函數。對這些函數,如果都用在命令行中指定源文件的方法同調用它們的程序鏈接起來,雖然也是可以的,但也有一些缺點:
.....
A如何生成靜態庫。
我們知道靜態庫也稱檔案庫,在此檔案文件中實際上是收集了一系列的目標文件。這些目標文件就是由cc(gcc)函數的源代碼編譯生成的。因此,靜態庫的生成方法實際上可以分成兩步:
1.將各函數代碼所在的源文件編譯成目錄文件。例如,對myfunc.c,可以用如下命令將其編譯成目標文件:
gcc -c myfunc.c
當然在有多個源文件時,只需在gcc 命令行中將其分別列上就可以了。
經此一步我休養將能夠得到各源文件的目標文件。對上例,將得到myfunc.o
2.將各目標文件收集起來放到一個靜態庫文件中。這主要借助ar命令完成,如:
ar r ~/lib/libtest.a myfunc.o
建立動態鏈接庫、並不需要用到其他的工具,借助gcc命令即可完成。此時需在命令
行中加上-K PIC和-G這兩個選項,如下我們可以建立libtest的動態版本:
gcc -K PIC -G -o $HOME/lib/libtest.so myfunc.c
------------------------------------------------------------
ldd filename就可以看到程序用到哪些庫。
ld -o OUTPUT /lib/crt0.o hello.o -lc
ld將輸出最終結果文件 OUTPUT ,
用到/lib/crt0.o hello.o以及libc.a
libc.a來自標準庫搜索路徑,請參看-l選項的討論
ld的選項順序任意,可以重復,面的將覆蓋前面的。
gcc -c -fPIC linuxlib.c -O3 -o linuxlib.o
gcc -shared -Wl,-soname,liblinuxlib.so.1 \
-O3 -o liblinuxlib.so.1.0 linuxlib.o
(
ld -m elf_i386 -shared -o liblinuxlib.so.1.0 \
-soname liblinuxlib.so.1 linuxlib.o
這個語句產生的文件更小,不知道二者有何區別
-o liblinuxlib.so.1.0 這個文件名任意,關鍵是面
兩個符號連接要正確
當然推薦使用有意義的帶版本信息的名字,包括
-soname liblinuxlib.so.1
)
ln -sf liblinuxlib.so.1.0 liblinuxlib.so.1
( 運行時,dynamic linker根據 -soname \
liblinuxlib.so.1 去尋找liblinuxlib.so.1
)
ln -sf liblinuxlib.so.1 liblinuxlib.so
( 編譯的最一個步驟,ld命令根據 -llinuxlib 選項去尋找liblinuxlib.so
)
export LD_LIBRARY_PATH=.
( 編譯時、運行時都需要這個變量的存在編譯時如果不存在這個變量,編譯依舊通過,但用ldd查看會發現沒有動態連接信息運行時如果不存在這個變量,將報告無法找到動態連接庫這個變量如果不用相對路徑".",換了運行環境就比較麻煩
export LD_LIBRARY_PATH=`pwd`:$LD_LIBRARY_PATH
)
gcc -L. -lbsd -llinuxlib -O3 -o linuxkiller linuxkiller.c
(
-L. 指出除標準庫搜索路徑之外的庫搜索路徑,如果不指定,將找不到liblinuxlib.so 這個選項並不能代替 LD_LIBRARY_PATH 變量,否則雖然編譯通過,但用ldd linuxkiller的時候會發現沒有動態連接信息
)
可以把自己的動態連接庫放到/usr/lib或者/lib下去,或者
修改/etc/ld.so.conf
然利用/sbin/ldconfig。具體請man ldconfig
---------------------------------------------------------
tt.c:
: #include <stdio.h>
: int func(){
: printf("func test\n");
: return 0;
: }
: ii.c:
: #include <stdio.h>
: extern int func();
: ...................
★ 生成靜態鏈接庫舉例
1. vi demo.h
void demo ( void );
vi demo.c
#include <stdio.h>
#include "demo.h"
void demo ( void )
{
printf( "hello world\n" );
return;
}
2. g++ -Wstrict-prototypes -Wall -Wunused -O3 -c demo.c -o demo.o
file demo.o
3. ar -rsv libdemo.a demo.o
a - demo.o
file libdemo.a
nm -s libdemo.a
ar -tv libdemo.a
4. vi scz.c
#include <stdio.h>
#include "demo.h"
int main ( int argc, void * argv[] )
{
demo();
return 0;
}
5. g++ scz.c -L. -ldemo -Wstrict-prototypes -Wall -Wunused -O3 -o scz
.....
A如何生成靜態庫。
我們知道靜態庫也稱檔案庫,在此檔案文件中實際上是收集了一系列的目標文件。這些目標文件就是由cc(gcc)函數的源代碼編譯生成的。因此,靜態庫的生成方法實際上可以分成兩步:
1.將各函數代碼所在的源文件編譯成目錄文件。例如,對myfunc.c,可以用如下命令將其編譯成目標文件:
gcc -c myfunc.c
當然在有多個源文件時,只需在gcc 命令行中將其分別列上就可以了。
經此一步我休養將能夠得到各源文件的目標文件。對上例,將得到myfunc.o
2.將各目標文件收集起來放到一個靜態庫文件中。這主要借助ar命令完成,如:
ar r ~/lib/libtest.a myfunc.o
建立動態鏈接庫、並不需要用到其他的工具,借助gcc命令即可完成。此時需在命令
行中加上-K PIC和-G這兩個選項,如下我們可以建立libtest的動態版本:
gcc -K PIC -G -o $HOME/lib/libtest.so myfunc.c
------------------------------------------------------------
ldd filename就可以看到程序用到哪些庫。
ld -o OUTPUT /lib/crt0.o hello.o -lc
ld將輸出最終結果文件 OUTPUT ,
用到/lib/crt0.o hello.o以及libc.a
libc.a來自標準庫搜索路徑,請參看-l選項的討論
ld的選項順序任意,可以重復,面的將覆蓋前面的。
gcc -c -fPIC linuxlib.c -O3 -o linuxlib.o
gcc -shared -Wl,-soname,liblinuxlib.so.1 \
-O3 -o liblinuxlib.so.1.0 linuxlib.o
(
ld -m elf_i386 -shared -o liblinuxlib.so.1.0 \
-soname liblinuxlib.so.1 linuxlib.o
這個語句產生的文件更小,不知道二者有何區別
-o liblinuxlib.so.1.0 這個文件名任意,關鍵是面
兩個符號連接要正確
當然推薦使用有意義的帶版本信息的名字,包括
-soname liblinuxlib.so.1
)
ln -sf liblinuxlib.so.1.0 liblinuxlib.so.1
( 運行時,dynamic linker根據 -soname \
liblinuxlib.so.1 去尋找liblinuxlib.so.1
)
ln -sf liblinuxlib.so.1 liblinuxlib.so
( 編譯的最一個步驟,ld命令根據 -llinuxlib 選項去尋找liblinuxlib.so
)
export LD_LIBRARY_PATH=.
( 編譯時、運行時都需要這個變量的存在編譯時如果不存在這個變量,編譯依舊通過,但用ldd查看會發現沒有動態連接信息運行時如果不存在這個變量,將報告無法找到動態連接庫這個變量如果不用相對路徑".",換了運行環境就比較麻煩
export LD_LIBRARY_PATH=`pwd`:$LD_LIBRARY_PATH
)
gcc -L. -lbsd -llinuxlib -O3 -o linuxkiller linuxkiller.c
(
-L. 指出除標準庫搜索路徑之外的庫搜索路徑,如果不指定,將找不到liblinuxlib.so 這個選項並不能代替 LD_LIBRARY_PATH 變量,否則雖然編譯通過,但用ldd linuxkiller的時候會發現沒有動態連接信息
)
可以把自己的動態連接庫放到/usr/lib或者/lib下去,或者
修改/etc/ld.so.conf
然利用/sbin/ldconfig。具體請man ldconfig
---------------------------------------------------------
tt.c:
: #include <stdio.h>
: int func(){
: printf("func test\n");
: return 0;
: }
: ii.c:
: #include <stdio.h>
: extern int func();
: ...................
★ 生成靜態鏈接庫舉例
1. vi demo.h
void demo ( void );
vi demo.c
#include <stdio.h>
#include "demo.h"
void demo ( void )
{
printf( "hello world\n" );
return;
}
2. g++ -Wstrict-prototypes -Wall -Wunused -O3 -c demo.c -o demo.o
file demo.o
3. ar -rsv libdemo.a demo.o
a - demo.o
file libdemo.a
nm -s libdemo.a
ar -tv libdemo.a
4. vi scz.c
#include <stdio.h>
#include "demo.h"
int main ( int argc, void * argv[] )
{
demo();
return 0;
}
5. g++ scz.c -L. -ldemo -Wstrict-prototypes -Wall -Wunused -O3 -o scz

