您的位置:首页 > 运维架构 > Linux

linux静态库、共享库、动态库三种库的创建和使用

2012-06-25 10:06 295 查看
参见文章

1.静态库

静态库就是一些目标文件的集合,以.a结尾。静态库在程序链接的时候使用,链接器会将程序中使用 到函数的代码从库文件中拷贝到应用程序中。一旦链接完成,在执行程序的时候就不需要静态库了。
gcc -c libhello -o libhello.o
ar rcs libhello.a libhello.o
链接静态库
gcc -c test.c -o test.o
gcc test.o -L. -lhello -o test
2.共享库

共享库以.so结尾. (so == share object) 在程序的链接时候并不像静态库那样在拷贝使用函数的代码,而只是作些标记。然后在程序开始启动运行的时候,动态地加载所需模块。

(1) 在所有基于GNU glibc的系统(当然包括Linux)中,在启动一个ELF二进制执行程序时, 一个特殊的程序"程序装载器"会被自动装载并运行。在linux中,这个程序装载器就是 /lib/ld-linux.so.X(X是版本号)。它会查找并装载应用程序所依赖的所有共享库。 被搜索的目录保存在/etc/ls.so.conf文件中,但一般/usr/local/lib并不在搜索之列。

(2) 当然你也可以通过设置环境变量LD_LIBRARY_PATH来设置ld的装载路径。这样装载器就会 首先搜索该变量的目录,然后才是默认目录。但是记住,LD_LIBRARY_PATH是用于开发和测试的,你可以将一些用于测试的替代共享库的目录放到该变量中,类似于/etc/ld.so.preload的作用。

但是该变量不应该用于正常用户的正常程序。

(3) 如果你不使用LD_LIBRARY_PATH环境变量,可以通过如下方式给装载器传入路径:

$ /lib/ld-linux.so.2 --library-path PATH EXECUTABLE

gcc -fPIC -g -c -Wall a.c
gcc -fPIC -g -c -Wall b.c
gcc -shared -Wl,-soname, libmyab.so.1 -o libmyab.so.1.0.1 a.o b.o -lpthread
一旦你创建好共享库后就需要安装使用了,最简单的办法是将库拷贝到默认目录下(/usr/lib)。 然后创建一些符号链接,最简单的方式还是使用ldconfig(8)来处理这里符号链接。最后是重新 编译链接你的程序,通过-L和-l参数指定库路径就可以了。

3.动态加载库

动态加载库(dynamically loaded (DL) libraries)是指在程序运行过程中可以加载的函数库。

而不是像共享库一样在程序启动的时候加载。DL对于实现插件和模块非常有用,因为他们可以让程序

在允许时等待插件的加载。在Linux中,动态库的文件格式跟共享库没有区别,主要区别在于共享库是运行时加载。

有专门的一组API用于完成打开动态库,查找符号,

处理出错,关闭动态库等功能。

下面对这些接口函数逐一介绍:

(1) dlopen

函数原型:void *dlopen(const char *libname,int flag);

功能描述:dlopen必须在dlerror,dlsym和dlclose之前调用,表示要将库装载到内存,准备使用。

如果要装载的库依赖于其它库,必须首先装载依赖库。如果dlopen操作失败,返回NULL值;如果库已经

被装载过,则dlopen会返回同样的句柄。

参数中的libname一般是库的全路径,这样dlopen会直接装载该文件;如果只是指定了库名称,在dlopen

会按照下面的机制去搜寻:

a.根据环境变量LD_LIBRARY_PATH查找

b.根据/etc/ld.so.cache查找

c.查找依次在/lib和/usr/lib目录查找。

flag参数表示处理未定义函数的方式,可以使用RTLD_LAZY或RTLD_NOW。RTLD_LAZY表示暂时不去

处理未定义函数,先把库装载到内存,等用到没定义的函数再说;RTLD_NOW表示马上检查是否存在未定义

的函数,若存在,则dlopen以失败告终。

(2) dlerror

函数原型:char *dlerror(void);

功能描述:dlerror可以获得最近一次dlopen,dlsym或dlclose操作的错误信息,返回NULL表示

无错误。dlerror在返回错误信息的同时,也会清除错误信息。

(3) dlsym

函数原型:void *dlsym(void *handle,const char *symbol);

功能描述:在dlopen之后,库被装载到内存。dlsym可以获得指定函数(symbol)在内存中的位置(指针)。

如果找不到指定函数,则dlsym会返回NULL值。但判断函数是否存在最好的方法是使用dlerror函数,

(4) dlclose

函数原型:int dlclose(void *);

功能描述:将已经装载的库句柄减一,如果句柄减至零,则该库会被卸载。如果存在析构函数,则在dlclose

之后,析构函数会被调用。

使用动态库
gcc -o dltest dltest.c -ldl -Wall
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: