歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
Linux教程網 >> Linux基礎 >> Linux服務器 >> Linux系統中靜態庫和動態庫的生成和使用

Linux系統中靜態庫和動態庫的生成和使用

日期:2017/3/2 16:49:34   编辑:Linux服務器

一、靜態庫的創建和使用:
1、生成靜態庫 :庫名 libmylib.a

ar rcs libmylib.a mylib.o

2、將靜態庫copy到 /usr/lib/ 或/lib/ 目錄下

cp libmylib.a /usr/lib/

3、靜態庫的使用

比如測試文件為test.c

gcc -0 test test.c -lmylib

-l為選項, mylib為庫名。mylib為libmylib的中間部分,Linux下約定所有庫都以前綴lib開始

靜態庫以.a結尾,動態庫以.so結尾。再編譯程式時,無需帶上前綴和後綴。

注意:靜態庫的命名需要以"lib"開頭,否者連接是編譯器無法找到庫

二、動態庫的創建和使用:

1、下面命令把mylib.c程序創建成了一個動態庫

(1)、gcc -fPIC -o mylib.o -c mylib.c

(2)、gcc -shared -o libttt.so mylib.o

也可以直接使用一條命令

gcc -fPIC -shared -o libttt.so mylib.c

2、Linux有兩種方式調用動態庫連接中的函數

(1)、gcc -o test test.c ./libttt.so

(2)、cp libttt.so /usr/lib/libttt.so

gcc -o test test.c /usr/lib/libttt.so

注意:引用動態庫時,必須含有路徑,如果只是使用libttt.so,則必須確保這個庫所在目錄包括再PATH 環境變量中

三、系統函數使用動態庫:

1、void *dlopen(const char *filename, int flag)

用於打開指定名字的動態鏈接庫,並返回一個句柄

flag:RTLD_LAZY, RTLD_NEW, RTLD_GLOBAL

RTLD_LAZY:在dlopen()返回前,對於動態庫中存在的未定義的變量(如外部變extern,也可以是函數)

不執行解析,也就是不解析這個變量的地址

RTLD_NEW:與RTLD_LAZY 不同,在dlopen()返回前,解析處每個未定義的變量的地址,如果解析不出來,dlopen會返回NULL,錯位為"undefined symbol:xxx..."

RTLD_GLOBAL:是庫中被解析出來的變量在隨後的其它鏈接庫中也可以使用,即全局有效。

2、void *dlsym(void *handle, char *symbol)

根據動態鏈接庫的句柄與函數名,返回函數名對應的函數的地址。

3、int dlclose(void *handle)

關閉動態鏈接庫,handle是調用dlopen函數庫的句柄

4、const char *dlerror(void)

動態庫鏈接庫執行失敗時,dlerror返回錯誤信息,若執行成功,則返回NULL

例子:

main.c
int main(void)
{
void *handle;
char *error;
void (*welcome)();

if ((handle = dlopen("./libttt.so", RTLD_LAZY)) == NULL)
{
printf("dlopen error\n");
return -1;
}

welcome = dlsym(handle, "welcome");
if ((error = dlerror()) != NULL)
{
printf("dlsym error\n");
return -1;
}
welcome();
dlclose(handle);

return 0;
}

gcc -ldl -o main main.c

-ldl 指明dlopen函數所在的庫。

注意:dlopen("./libttt.so", RTLD_LAZY),的目錄是在當前目錄下,如果不在當前目錄下,程序運行錯誤

共享庫進行更新或安裝新庫後,必須運行ldconfig命令更新/etc/ld.conf文件中相應的項

如果使用RPM進行安裝,一般會自動進行更新,不過也不能保證這一點。

使用數學庫時加上 -lm

gcc -lm

庫工具使用:

ldd 工具

ldd 用來顯示執行文件需要哪些共享庫, 共享庫裝載管理器在哪裡找到了需要的共享庫

Copyright © Linux教程網 All Rights Reserved