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

linux动态库与静态库的编译与加载

2015-08-24 09:16 786 查看
http://blog.163.com/xychenbaihu@yeah/blog/static/13222965520101023104745738/

一、基本概念

1.1、什么是库

在windows平台和linux平台下都大量存在着库。

本质上来说库是一种可执行的二进制代码(但不可以独立执行),可以被操作系统载入内存执行

由于windows和linux的平台不同(主要是编译器、汇编器和连接器的不同),因此二者库的二进制是不兼容的。

本文仅限于介绍linux下的库。

1.2、库的种类

linux下的库有两种:静态库和共享库(动态库)。

二者的不同点在于代码被载入的时刻不同:

静态库的代码在编译过程中已经被载入可执行程序,因此生成的可执行程序体积较大。静态用.a为后缀,
例如:libhello.a

共享库(动态库)的代码是在可执行程序运行时才载入内存的,在编译过程中仅简单的引用,因此生成的可执行程序代码体积较小。

动态通常用.so为后缀,例如:libhello.so

共享库(动态库)的好处是::不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例。

为了在同一系统中使用不同版本的库,可以在库文件名后加上版本号为后缀,例如:libhello.so.1.0,由于程序连接默认以.so为文件后缀名。所以为了使用这些库,通常使用建立符号连接的方式。

ln-slibhello.so.1.0libhello.so.1

ln-slibhello.so.1libhello.so

1.3、静态库,动态库文件在linux下是如何生成的:

以下面的代码为例,生成上面用到的hello库:

/*hello.c*/

#include"hello.h"
voidsayhello()
{
printf("hello,world");
}
首先用gcc编绎该文件,在编绎时可以使用任何合法的编绎参数,例如-g加入调试代码等:
gcc-chello.c-ohello.o
1、生成静态库生成静态库使用ar工具,其实ar是archive的意思
arcqslibhello.ahello.o
2、生成动态库用gcc来完成,由于可能存在多个版本,因此通常指定版本号:gcc-shared-olibhello.so.1.0hello.o
1.4、库文件是如何命名的,有没有什么规范:
在linux下,库文件一般放在/usr/lib和/lib下,
静态库的名字一般为libxxxx.a,其中xxxx是该lib的名称;
动态库的名字一般为libxxxx.so.major.minor,xxxx是该lib的名称,major是主版本号,minor是副版本号
1.5、可执行程序在执行的时候如何定位共享库(动态库)文件:
当系统加载可执行代码(即库文件)的时候,能够知道其所依赖的库的名字,但是还需要知道绝对路径,此时就需要系统动态载入器(dynamiclinker/loader)
对于elf格式的可执行程序,是由ld-linux.so*来完成的,它先后搜索elf文件的DT_RPATH段-->环境变量LD_LIBRARY_PATH—->/etc/ld.so.cache文件列表-->/lib/,/usr/lib目录找到库文件后将其载入内存
如:exportLD_LIBRARY_PATH=’pwd’
将当前文件目录添加为共享目录。
1.6、使用ldd工具,查看可执行程序依赖那些动态库或着动态库依赖于那些动态库:
ldd命令可以查看一个可执行程序依赖的共享库,
例如#ldd/bin/lnlibc.so.6
=>/lib/libc.so.6(0×40021000)/lib/ld-linux.so.2
=>/lib/ld-linux.so.2(0×40000000)
可以看到ln命令依赖于libc库和ld-linux库
1.7、使用nm工具,查看静态库和动态库中有那些函数名;
(T类表示函数是当前库中定义的,U类表示函数是被调用的,在其它库中定义的,W类是当前库中定义,被其它库中的函数覆盖)。:
有时候可能需要查看一个库中到底有哪些函数,nm工具可以打印出库中的涉及到的所有符号,这里的库既可以是静态的也可以是动态的。nm列出的符号有很多,常见的有三种::T类:是在库中定义的函数,用T表示,这是最常见的;U类:是在库中被调用,但并没有在库中定义(表明需要其他库支持),用U表示;W类:是所谓的“弱态”符号,它们虽然在库中被定义,但是可能被其他库中的同名符号覆盖,用W表示。例如,假设开发者希望知道上文提到的hello库中是否引用了printf():
nmlibhello.so|grepprintf
发现printf是U类符号,说明printf被引用,但是并没有在库中定义。由此可以推断,要正常使用hello库,必须有其它库支持,使用ldd工具查看hello依赖于哪些库:lddlibhello.solibc.so.6=>/lib/libc.so.6(0x400la000)/lib/ld-linux.so.2=>/lib/ld-linux.so.2(0x40000000)从上面的结果可以继续查看printf最终在哪里被定义,有兴趣可以goon1.8、使用ar工具,可以生成静态库,同时可以查看静态库中包含那些.o文件,即有那些源文件构成。可以使用ar-tlibname.a来查看一个静态库由那些.o文件构成。可以使用arqlibname.axxx1.oxxx2.oxxx3.o...xxxn.o生成静态库1.9、如何查看动态库和静态库是32位,还是64位下的库:如果是动态库,可以使用file*.so;如果是静态哭,可以使用objdump-x*.a
Linux下进行程序设计时,关于库的使用:
一、gcc/g++命令中关于库的参数:
-shared:该选项指定生成动态连接库;
-fPIC:表示编译为位置独立(地址无关)的代码,不用此选项的话,编译后的代码是位置相关的,所以动态载入时,是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。
-L:指定链接库的路径,-L.表示要连接的库在当前目录中
-ltest:指定链接库的名称为test,编译器查找动态连接库时有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称
-Wl,-rpath:记录以来so文件的路径信息。
LD_LIBRARY_PATH:这个环境变量指示动态连接器可以装载动态库的路径。
当然如果有root权限的话,可以修改/etc/ld.so.conf文件,然后调用/sbin/ldconfig来达到同样的目的,
不过如果没有root权限,那么只能采用修改LD_LIBRARY_PATH环境变量的方法了。
调用动态库的时候,有几个问题会经常碰到:
1、有时,明明已经将库的头文件所在目录通过“-I”include进来了,库所在文件通过“-L”参数引导,并指定了“-l”的库名,但通过ldd命令察看时,就是死活找不到你指定链接的so文件,这时你要作的就是通过修改LD_LIBRARY_PATH或者/etc/ld.so.conf文件来指定动态库的目录。通常这样做就可以解决库无法链接的问题了。
二、静态库链接时搜索路径的顺序:
1.ld会去找gcc/g++命令中的参数-L;
2.再找gcc的环境变量LIBRARY_PATH,它指定程序静态链接库文件搜索路径;
exportLIBRARY_PATH=$LIBRARY_PATH:data/home/billchen/lib
3.再找默认库目录/lib/usr/lib/usr/local/lib,这是当初compilegcc时写在程序内的。
三、动态链接时、执行时搜索路径顺序:
1.编译目标代码时指定的动态库搜索路径;
2.环境变量LD_LIBRARY_PATH指定动态库搜索路径,它指定程序动态链接库文件搜索路径;
exportLD_LIBRARY_PATH=$LD_LIBRARY_PATH:data/home/billchen/lib
3.配置文件/etc/ld.so.conf中指定的动态库搜索路径;
4.默认的动态库搜索路径/lib;
5.默认的动态库搜索路径/usr/lib。
四、静态库和动态链接库同时存在时,gcc/g++默认链接的是动态库:
当一个库同时存在静态库和动态库时,比如libmysqlclient.a和libmysqlclient.so同时存在时:
在Linux下,动态库和静态库同事存在时,gcc/g++的链接程序,默认链接的动态库

可以使用下面的方法,给连接器ld传递参数,看是否链接动态库还是静态库。

-Wl,-Bstatic-llibname//指定让gcc/g++链接静态库
使用:
gcc/g++test.c-otest-Wl,-Bstatic-llibname-Wl,-Bdynamic-lm-lc

-Wl,-Bdynamic-llibname//指定让gcc/g++链接动态库
使用:
gcc/g++test.c-otest-Wl,-Bdynamic-llibname
如果要完全静态加在,使用-static参数,即将所有的库以静态的方式链入可执行程序,这样生成的可执行程序,不再依赖任何库,同事出现的问题是,这样编译出来的程序非常大,占用空间。
如果不适用-Wl,-Bdynamic-lm-c会有如下错误:
[chenbaihu@build17lib]$ls
libtest.alibtest.sott.cctest.cctest.htest.o
[chenbaihu@build17lib]$g++-Wall-gt.cc-ot-L./-Wl,-Bstatic-ltest-Wl,-Bdynamic-lm-lc
[chenbaihu@build17lib]$g++-Wall-gt.cc-ot-L./-Wl,-Bstatic-ltest
/usr/bin/ld:cannotfind-lm
collect2:ld返回1


五、Linux下动态库为什么会出现编译OK,运行时找不到的情况。


原因是:

linux下链接器默认是不记录库的搜索路径的,只记录名字,所以才会有编译时OK,但运行时,找不到的情况。

解决方法:

想在程序中记录路径,可以使用-WI,-rlibpath指定动态库的搜索路径。

使用方法。

gcc/g++test.c-otest-WI,rlibpath-llibname
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: