linux下的静态库与动态库的区别和使用.doc_第1页
linux下的静态库与动态库的区别和使用.doc_第2页
linux下的静态库与动态库的区别和使用.doc_第3页
linux下的静态库与动态库的区别和使用.doc_第4页
linux下的静态库与动态库的区别和使用.doc_第5页
免费预览已结束,剩余1页可下载查看

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1. Linux动态库和静态库的区别1.什么是库在windows平台和linux平台下都大量存在着库。本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执行。由于windows和linux的本质不同,因此二者库的二进制是不兼容的。本文仅限于介绍linux下的库。2.库的种类linux下的库有两种:静态库和共享库(动态库)。二者的不同点在于代码被载入的时刻不同。静态库在程序编译时会被连接到目标代码中,目标程序运行时将不再需要该动态库,移植方便,体积较大,但是浪费空间和资源,因为所有相关的对象文件与牵涉到的库被链接合成一个可执行文件。动态库在程序编译时并不会被连接到目标代码中,而是在程序运行时才被载入,因此体积较小,可以实现进程间的资源共享,甚至可以真正做到链接载入完全由程序员在程序代码中控制,另外将一些程序的升级变得简单,但是在程序运行时需要动态库存在。3.库存在的意义库是别人写好的现有的,成熟的,可以复用的代码,你可以使用但要记得遵守许可协议。现实中每个程序都要依赖很多基础的底层库,不可能每个人的代码都从零开始,因此库的存在意义非同寻常。共享库的好处是,不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的实例。4.库文件是如何产生的在linux下静态库的后缀是.a,它的产生分两步Step 1.由源文件编译生成一堆.o,每个.o里都包含这个编译单元的符号表Step 2.ar命令将很多.o转换成.a,成文静态库例如:$gcc -c hello.c$ ar crs libmyhello.a hello.o动态库的后缀是.so,它由gcc加特定参数编译产生。例如:$gcc -c hello.c$ gcc -shared -fPIC -o libmyhello.so hello.o参数解析:-shared:指定生成动态链接库-static:指定生成静态链接库-fPIC:表示编译为位置独立的代码,用于编译共享库。目标文件需要创建成位置无关码概念上就是在可执行程序装载它们的时候,它们可以放在可执行程序的内存里的任何地方 -l:指定链接时需要动态库,编译器查找动态链接库有隐含的命名规则,即在给出的名字前面加上lib,后面加上.so来确定库的名称。-ggdb:此选项将尽可能的生成gdb的可以使用的调试信息- g:编译器在编译的时候产生调试信息-c:只激活预处理、编译和汇编,也就是把程序做成目标文件(.o文件)。-Wl,options:把参数(options)传递给连接器ld。如果options中间有逗号,就将options分成多个选项,然后传递给链接程序 -L.:表示要连接的库在当前目录中-Wall:生成所有警告信息5.库文件是如何命名的,有没有什么规范在linux下,库文件一般放在/usr/lib /lib下,静态库的名字一般为libxxxx.a,其中xxxx是该lib的名称,动态库的名字一般为libxxxx.so.major.minor,xxxx是该lib的名称,major是主版本号, minor是副版本号6.如何知道一个可执行程序依赖哪些库ldd命令可以查看一个可执行程序依赖的共享库,例如# ldd /bin/lnlibc.so.6= /lib/libc.so.6 (040021000)/lib/ld-linux.so.2= /lib/ld- linux.so.2 (040000000)可以看到ln命令依赖于libc库和ld-linux库7.可执行程序在执行的时候如何定位动态库文件当系统加载可执行代码时候,能够知道其所依赖的库的名字,但是还需要知道绝对路径此时就需要系统动态载入器(dynamic linker/loader)对于elf格式的可执行程序,是由ld-linux.so*来完成的,它先后搜索elf文件的 DT_RPATH段 - 环境变量LD_LIBRARY_PATH /etc/ld.so.cache文件列表 - /lib/和/usr/lib目录找到库文件后将其载入内存为了让执行程序顺利找到动态库,有三种方法:(1)把库拷贝到/usr/lib和/lib目录下, 那么ld默认能够找到。(2)在LD_LIBRARY_PATH环境变量中加上库所在路径。 例如动态库libhello.so在/home/ting/lib目录下,以bash为例,使用命令: $export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/ting/lib (3) 如果安装在其他目录,需要将其添加到/etc/ld.so.cache文件中,步骤如下 1.编辑/etc/ld.so.conf文件,加入库文件所在目录的路径2.运行ldconfig,该命令会重建/etc/ld.so.cache文件8.在新安装一个库之后如何让系统能够找到他8.查看库中的符号 有时候可能需要查看一个库中到底有哪些函数,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) 9.举例来说明在Linux中如何创建静态库和动态库,以及使用它们。在创建函数库前,我们先来准备举例用的源程序,并将函数库的源程序编译成.o文件。 第1步:编辑得到举例的程序-hello.h、hello.c和main.c; hello.h(见程序1)为该函数库的头文件。hello.c(见程序2)是函数库的源程序,其中包含公用函数hello,该函数将在屏幕上输出Hello XXX!。main.c(见程序3)为测试库文件的主程序,在主程序中调用了公用函数hello。 程序1: hello.h#ifndef HELLO_H#define HELLO_Hvoid hello(const char *name);#endif /HELLO_H程序2: hello.c#include void hello(const char *name)printf(Hello %s!n, name);程序3: main.c#include hello.hint main()hello(everyone);return 0;第2步:将hello.c编译成.o文件; 无论静态库,还是动态库,都是由.o文件创建的。因此,我们必须将源程序hello.c通过gcc先编译成.o文件。 在系统提示符下键入以下命令得到hello.o文件。 # gcc -c hello.c (注1:首字符#是系统提示符,不需要键入,下文相同。) # ls (注2:在ls命令结果中,我们看到了hello.o文件,本步操作完成) hello.c hello.h hello.o main.c # 第3步:由.o文件创建静态库; 静态库文件名的命名规范是以lib为前缀,紧接着跟静态库名,扩展名为.a。例如:我们将创建的静态库名为myhello,则静态库文件名就是libmyhello.a。在创建和使用静态库时,需要注意这点。创建静态库用ar命令。 在系统提示符下键入以下命令将创建静态库文件libmyhello.a。 # ar crs libmyhello.a hello.o# ls (注:ls命令结果中有libmyhello.a)hello.c hello.h hello.o libmyhello.a main.c# 第4步:在程序中使用静态库; 静态库制作完了,如何使用它内部的函数呢?只需要在使用到这些公用函数的源程序中包含这些公用函数的原型声明,然后在用gcc命令生成目标文件时指明静态库名,gcc将会从静态库中将公用函数连接到目标文件中。注意,gcc会在静态库名前加上前缀lib,然后追加扩展名.a得到的静态库文件名来查找静态库文件。在程序3:main.c中,我们包含了静态库的头文件hello.h,然后在主程序main中直接调用公用函数hello。下面先生成目标程序hello,然后运行hello程序看看结果如何。 # gcc -o hello main.c -L. -lmyhello # ./hello Hello everyone!# 我们删除静态库文件试试公用函数hello是否真的连接到目标文件 hello中了。# rm libmyhello.arm: remove regular file libmyhello.a? y# ./helloHello everyone!# 程序照常运行,静态库中的公用函数已经连接到目标文件中了。 我们继续看看如何在Linux中创建动态库。我们还是从.o文件开始。 第5步:由.o文件创建动态库文件 动态库文件名命名规范和静态库文件名命名规范类似,也是在动态库名增加前缀lib,但其文件扩展名为.so。例如:我们将创建的动态库名为myhello,则动态库文件名就是libmyhello.so。用gcc来创建动态库。 在系统提示符下键入以下命令得到动态库文件libmyhello.so。 # gcc -shared -fPIC -o libmyhello.so hello.o # lshello.c hello.h hello.o libmyhello.so main.c# 第6步:在程序中使用动态库 在程序中使用动态库和使用静态库完全一样,也是在使用到这些公用函数的源程序中包含这些公用函数的原型声明,然后在用gcc命令生成目标文件时指明动态库名进行编译。我们先运行gcc命令生成目标文件,再运行它看看结果。 # gcc -o hello main.c -L. -lmyhello# ./hello./hello: error while loading shared libraries: libmyhello.so: cannot open shared object file: No such file or directory #哦!出错了。快看看错误提示,原来是找不到动态库文件libmyhello.so。程序在运行时,会在/usr/lib和/lib等目录中查找需要的动态库文件。若找到,则载入动态库,否则将提示类似上述错误而终止程序运行。我们将文件libmyhello.so复制到目录/usr/lib中,再试试。 # mv libmyhello.so /usr/lib# ./helloHello everyone!# 成功了。这也进一步说明了动态库在程序运行时是需要的。第7步:动态库和静态库重名问题 我们回过头看看,发现使用静态库和使用动态库编译成目标程序使用的gcc命令完全一样,那当静态库和动态库同名时,gcc命令会使用哪个库文件呢?抱着对问题必究到底的心情,来试试看. 先删除 除.c和.h外的 所有文件,恢复成我们刚刚编辑完举例程序状态.# rm -f hello hello.o /usr/lib/libmyhello.so# ls hello.c hello.h main.c#在来创建静态库文件libmyhello.a和动态库文件libmyhello.so# gcc -c hello.c # ar crs libmyhello.a hello.o# gcc -shared -fPCI -o libmyhello.so hello.o# lshello.c hello.h hello.o libmyhello.a libmyhello.so main.c#通过上述最后一条ls命令,可以发现静态库文件libmyhello.a和动态库文件libmyhello.so都已经生成,并都

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论