长春网站建设免费咨询,几大门户网站,网站 参数设置,wordpress素材关于Unix静态库和动态库的分析 基本概念 库有动态与静态两种#xff0c;动态通常用.so为后缀#xff0c;静态用.a为后缀。 例如#xff1a;libhello.so libhello.a 为了在同一系统中使用不同版本的库#xff0c;可以在库文件名后加上版本号为后缀,例如#xff1a; libhell… 关于Unix静态库和动态库的分析 基本概念 库有动态与静态两种动态通常用.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 1、使用库 当要使用静态的程序库时连接器会找出程序所需的函数然后将它们拷贝到执行文件由于这种拷贝是完整的所以一旦连接成功静态程序库也就不再需要了。 然而对动态库而言就不是这样。动态库会在执行程序内留下一个标记指明当程序执行时首先必须载入这个库。由于动态库节省空间linux下进行连接的 缺省操作是首先连接动态库也就是说如果同时存在静态和动态库不特别指定的话将与动态库相连接。 现在假设有一个叫hello的程序开发包它提供一个静态库libhello.a 一个动态库libhello.so,一个头文件hello.h,头文件中提供sayhello()这个函数 /* hello.h */ void sayhello(); 另外还有一些说明文档。 这一个典型的程序开发包结构 与动态库连接 linux默认的就是与动态库连接下面这段程序testlib.c使用hello库中的sayhello()函数 /*testlib.c*/ #include hello.h int main() { sayhello(); return 0; } 使用如下命令进行编译 $gcc -c testlib.c -o testlib.o 用如下命令连接 $gcc testlib.o -lhello -o testlib 连接时要注意假设libhello.so 和libhello.a都在缺省的库搜索路径下/usr/lib下如果在其它位置要加上-L参数。与静态库连接麻烦一些主要是参数问题。还是上面的例 子 $gcc testlib.o -o testlib -WI,-Bstatic -lhello 注这个特别的”-WI,-Bstatic”参数实际上是传给了连接器ld指示它与静态库连接如果系统中只有静态库当然就不需要这个参数了。如果要和多个库相连接而每个库的连接方式不一样比如上面的程序既要和 libhello进行静态连接又要和libbye进行动态连接其命令应为 $gcc testlib.o -o testlib -WI,-Bstatic -lhello -WI,-Bdynamic -lbye 注意: -WI,-Bstatic -l库名 //如果动态库和静态库同时存在则加载静态库。 -WI,Bdynamic -l库名 //如果动态库和静态库同时存在则加载动态库。 2、动态库的路径问题 //指定库的路径的方法 为了让执行程序顺利找到动态库有三种方法 把库拷贝到/usr/lib和/lib目录下。在LD_LIBRARY_PATH环境变量中加上库所在路径。例如动态库 libhello.so在/home/ting/lib目录下以bash为例使用命令 $export LD_LIBRARY_PATH$LD_LIBRARY_PATH:/home/ting/lib修改/etc/ld.so.conf文件把库所在的路径加到文件末尾并执行ldconfig刷新。这样加入的目录下的所有库文件都可见。 注意: 也可以在使用gcc/g编译程序时让生成的可执行程序记住动态库的位置方法: gcc/g test.c -o test -WI,rlibpath -llibname 这样编译出来的程序自己就可以记住库的路径就可以动态加载了。 3、查看库中的符号 有时候可能需要查看一个库中到底有哪些函数nm工具可以打印出库中的涉及到的所有符号。库既可以是静态的也可以是动态的。nm列出的符号有很多 常见的有三种一种是在库中被调用但并没有在库中定义(表明需要其他库支持)用U表示一种是库中定义的函数用T表示这是最常见的另外一种是所 谓的“弱态”符号它们虽然在库中被定义但是可能被其他库中的同名符号覆盖用W表示。例如假设开发者希望知道上文提到的hello库中是否引用了 printf(): $nm libhello.so | grep printf U 其中printf U表示符号printf被引用但是并没有在函数内定义由此可以推断要正常使用hello库必须有其它库支持再使用ldd工具查看hello依赖于哪些库 $ldd hello libc.so.6/lib/libc.so.6(0x400la000) /lib/ld-linux.so.2/lib/ld-linux.so.2 (0x40000000) 从上面的结果可以继续查看printf最终在哪里被定义有兴趣可以go on 可以使用 ar -t libname.a 来查看 4、生成库 第一步要把源代码编绎成目标代码。以下面的代码为例生成上面用到的hello库 /* hello.c */ #include hello.h void sayhello() { printf(hello,world ); } 用gcc编绎该文件在编绎时可以使用任何合法的编绎参数例如-g加入调试代码等 $gcc -c hello.c -o hello.o 1.连接成静态库 连接成静态库使用ar工具其实ar是archive的意思 $ar cqs libhello.a hello.o 2.连接成动态库 生成动态库用gcc来完成由于可能存在多个版本因此通常指定版本号 $gcc -shared -Wl,-soname,libhello.so.1 -o libhello.so.1.0 hello.o 另外再建立两个符号连接 $ln -s libhello.so.1.0 libhello.so.1 $ln -s libhello.so.1 libhello.so 这样一个libhello的动态连接库就生成了。最重要的是传gcc -shared 参数使其生成是动态库而不是普通执行程序。 -Wl 表示后面的参数也就是-soname,libhello.so.1直接传给连接器ld进行处理。实际上每一个库都有一个soname当连接器发现它正 在查找的程序库中有这样一个名称连接器便会将soname嵌入连结中的二进制文件内而不是它正在运行的实际文件名在程序执行期间程序会查找拥有 soname名字的文件而不是库的文件名换句话说soname是库的区分标志。这样做的目的主要是允许系统中多个版本的库文件共存习惯上在命名库 文件的时候通常与soname相同 libxxxx.so.major.minor 其中xxxx是库的名字major是主版本号minor 是次版本号 转载于:https://www.cnblogs.com/wangfengju/archive/2013/01/28/6173294.html