动态链接库及静态链接库(
动态链接库及静态链接库(Windows下的.dll .Iib 和IinuX下的.so .a )
库有动态与静态两种,动态通常用.so为后缀,静态用.a为后缀。例如:libhello.so libhello.a
为了在冋一系统中使用不冋版本的库,可以在库文件名后加上版本号为后缀
,例如:
libhello.so.1.0,由于程序连接默认以.so为文件后缀名。所以为了使用这些库,
通常使用建立
符号连接的方式。
ln -S libhello.so.1.0 libhello.so.1
ln -S libhello.so.1 libhello.so
使用库
当 要使用静态的程序库时,连接器会找出程序所需的函数,然后将它们拷贝到执行文件,
由于这种拷贝是完整的,所以一旦连接成功,静态程序库也就不再需要了。然而,对动态
库而言,就不是这样。动态库会在执行程序内留下一个标记,指明当程序执行时,首先必须
载入这个库。由于动态库节省空间,linUX下进行连接的缺省操作是首先连接动态库,也就
是说,如果同时存在静态和动态库,不特别指定的话,将与动态库相连接。
现在假设有一个叫hello的程序开发包,它提供一个静态库libhello.a 一个动态库libhello.so,
一个头文件hello.h,头文件中提供Sayhello()这个函数
/* hello.h */
void Sayhello();
另外还有一些说明文档。这一个典型的程序开发包结构
与动态库连接]
linux默认的就是与动态库连接,下面这段程序testlib.c使用hello库中的Sayhello()函数
/*testlib.c*/
#i nclude
#i nclude
int mai n()
{
Sayhello();
return 0;
}
使用如下命令进行编译
$gcc -C testlib.c -o testlib.o
用如下命令连接:
$gcc testlib.o -lhello -o testlib
在连接时要注意,假设libhello.o和libhello.a都在缺省的库搜索路径下/usr/lib下,如果在
其匕位置要加上-L参数
与与静态库连接麻烦一些,主要是参数问题。还是上面的例子:
$gcc testlib.o -o testlib -WI,-Bstatic -lhello
注:这个特别的"-WI , -Bstatic"参数,实际上是传给了连接器Id.
指示它与静态库连接,如果系统中只有静态库当然就不需要这个参数了。
如果要和多个库相连接,而每个库的连接方式不一样,比如上面的程序既要和libhello进行
静态连接,又要和Iibbye进行动态连接,其命令应为:
$gcc testlib.o -o testlib -WI,-Bstatic -lhello -Wl,-Bdy namic -lbye
3.动态库的路径问题
为了让执行程序顺利找到动态库,有三种方法:
-Wl表示后面的参数也就是
-Wl表示后面的参数也就是-Soname,libhello.so.1直接传给连接器ld进行处理。实际上,每
把库拷贝到∕usr∕lib和/Iib目录下。
⑵ 在 LD_LIBRARY_PATH 环境变量中加上库所在路径。例如 动态库IibheIlo.so 在 ∕home∕ting/lib目录下,以 bash为例,使用命令:|
$export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/ti ng/lib
(3)修改/etc/ld.so.conf文件,把库所在的路径加到文件末尾,并执行ldconfig刷新。这样,
加入的目录下的所有库文件都可见、
4.查看库中的符号
有 时候可能需要查看一个库中到底有哪些函数,nm命令可以打印出库中的涉及到的所有
符号。库既可以是静态的也可以是动态的。nm列出的符号有很多,常见的有 三种,一种
是在库中被调用,但并没有在库中定义(表明需要其他库支持),用U表示;一种是库中定义 的函数,用T表示,这是最常见的;另外一种是所谓的弱 态”符号,它们虽然在库中被定
义,但是可能被其他库中的同名符号覆盖,用W表示。例如,假设开发者希望知道上央提
到的hello库中是否定义了 Printf():
$nm libhello.so |grep Printf
U Printf
U表示符号Printf被引用,
但是并没有在函数内定义,由此可以推断,要正常使用hello库,
必须有其它库支持,再使用
ldd命令查看hello依赖于哪些库:
$ldd hello
Iibc.so?6=>∕lib∕libc?so?6(0x400la000)
∕lib∕ld-lin
ux.so.2=