`
makeyouown
  • 浏览: 50475 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

linux下动态库和静态库引用问题解决

 
阅读更多
基本概念
库有动态与静态两种,动态通常用.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、动态库的路径问题                                            //指定库的路径的方法
为了让执行程序顺利找到动态库,有三种方法:

1.把库拷贝到/usr/lib和/lib目录下。
2.在LD_LIBRARY_PATH环境变量中加上库所在路径。例如动态库 libhello.so在/home/ting/lib目录下,以bash为例,使用命令: $export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/ting/lib
3.修改/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.o1.连接成静态库 连接成静态库使用ar工具,其实ar是archive的意思

$ar cqs libhello.a hello.o2.连接成动态库 生成动态库用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 是次版本号

总结
通过对LINUX库工作的分析,我们已经可以理解程序运行时如何去别的地方寻找“库”。

附上针对这个工程的Makefile:

# xiejingquan@gmail.com  # export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:~/C/lib    BIN_DIR=bin  LIB_DIR=lib  INC_DIR=inc  SRC_DIR=src    BIN=${BIN_DIR}/testlib  LIB=${LIB_DIR}/libhello.a ${LIB_DIR}/libhello.so    CC=gcc  AR=ar  DOC=doxygen    CFLAGS=-g -Wall -c -Iinc  LFLAGS=-lhello -L${LIB_DIR} -o  ARFLAGS=cqs  SOFLAGS=-shared -Wl,-soname,    all: ${BIN}    lib: ${LIB}    run: all  @./bin/testlib    clean:  rm -f ${BIN_DIR}/* ${LIB_DIR}/*    ${BIN_DIR}/testlib: ${LIB_DIR}/testlib.o ${LIB}  ${CC} $< ${LFLAGS} $@    ${LIB_DIR}/testlib.o: ${SRC_DIR}/testlib.c ${INC_DIR}/hello.h  ${CC} ${CFLAGS} $< -o $@    ${LIB_DIR}/libhello.a: ${LIB_DIR}/hello.o  ${AR} ${ARFLAGS} $@ $<    ${LIB_DIR}/libhello.so: ${LIB_DIR}/hello.o  ${CC} ${SOFLAGS}libhello.so.1 -o ${LIB_DIR}/libhello.so.1.0 ${LIB_DIR}/hello.o  ln -s ${LIB_DIR}/libhello.so.1.0 ${LIB_DIR}/libhello.so.1  ln -s ${LIB_DIR}/libhello.so.1 ${LIB_DIR}/libhello.so    ${LIB_DIR}/hello.o: ${SRC_DIR}/hello.c ${INC_DIR}/hello.h  ${CC} ${CFLAGS} $< -o $@附上文件的目录结构:

|– bin
|   `– testlib
|– doc
|– inc
|   `– hello.h
|– lib
|   |– hello.o
|   |– libhello.a
|   |– libhello.so -> lib/libhello.so.1
|   |– libhello.so.1 -> lib/libhello.so.1.0
|   |– libhello.so.1.0
|   `– testlib.o
|– src
|   |– hello.c
|   `– testlib.c


--------------------------------------------------------------------------------

分享到:
评论

相关推荐

    grpc-1.30.2 Linux 编译静动态库

    默认编译是静态库,但考虑到 linux 上动态库使用较多,所以使用 -DBUILD_SHARED_LIBS=ON 参数编译为动态库。 在 centos 7 下使用 gcc 4.8.5 + cmake 3.16.9 编译,包含 bin, include, lib, lib64, share 五个目录,...

    Linux静态库和动态库的分析及实现

     linux下的库有两种:静态库和共享库(动态库)。  二者的不同点在于代码被载入的时刻不同。  静态库的代码在编译过程中已经被载入可执行程序,因此体积较大。  共享库的代码是在可执行程序运行时才载入内存...

    gRPC-1.62.1 Linux 编译静动态库

    默认编译是静态库,但考虑到 linux 上动态库使用较多,所以使用 -DBUILD_SHARED_LIBS=ON 参数编译为动态库。 在 ubuntu 下使用 gcc 4.8.5 + cmake 3.28 编译,包含 bin, include, lib, lib64, share 五个目录,可以...

    Linux下动态连接.doc

    Linux动态连接 Linux中的应用程序以以下两种方式之一链接到外部函数 -&gt; 在构建时与静态库(lib*.a)静态链接,并且将库代码包含在该应用程序的可执行文件里 -&gt; 在运行时与共享库(lib*.so)动态链接,通过动态链接装入...

    linux C库文件的建立和使用资料

    linux下的库有两种:静态库和共享库(动态库)。 二者的不同点在于代码被载入的时刻不同。 静态库的代码在编译过程中已经被载入可执行程序,因此体积较大。 共享库的代码是在可执行程序运行时才载入内存的,在编译...

    讯飞linux_sdk.rar

    5.bin目录下存放SDK需要调用的动态库(动态库版本才会提供)、命令词列表样例、abnf语法样例、符合标准的语音文件样例、配置文件等,make之后的example可执行程序也会拷贝至此,请在此目录下运行,否则会运行失败;...

    为Linux应用程序编写DLL程序函数

    Linux 中的应用程序以以下两种方式之一链接到外部函数:要么在构建时与静态库( lib*.a)静态地链接,并且将库代码包含在该应用程序的可执行文件里;要么在运行时与共享库( lib*.so)动态地链接。通过动态链接装入...

    PowerPC上ELF可执行文件的符号解析(二)

    符号解析是Linux系统导入二进制可执行文件的重要过程,它完成的工作包括将一个符号定位到实际...我们可以这么认为,如果一个符号在共享库中定义,那么当其他可执行文件或共享库引用这个符号时,就需要对它作动态解析。

    传智播客扫地僧视频讲义源码

    06_动态库测试环境和调试环境搭建 07_socketclient动态库函数_初始化实现wmv 08_socketclient动态库函数_发送接受释放实现wmv 09_socketclient动态库_第二套api函数_传智扫地僧 10_socketclient动态库_日志功能集成 ...

    VC中LINK 2001 和 LINK 2009 的错误的解决方法

    首先,关于VC中的lib,与linux下的静态库是不同的,在VC中编译动态库的时候会生成一个lib和一个对应的dll,使用者在使用的时候需要包含头文件以及连接到该lib,在发布最终程序的时候则需要将对应的dll拷贝到发布目录...

    链接器和加载器.PDF(链接器和加载器 Beta 2)

    《链接器和加载器》讲述构建程序的关键工具——链接器和加载器,内容包括链接和加载、体系结构、目标文件、存储分配、符号管理、库、重定位、加载和覆盖、共享库、动态链接和加载、动态链接的共享库,以及着眼于成熟...

    易IDE插件支持库1.2版(eIDE.fne)-易语言

    代码管理,模块导入,编译辅助:静态编译链接器选择和动态、静态、黑月编译链接后处理。 使用说明:1、代码管理右键菜单可增加、删除及修改已存储代码 2、模块引用列表可设置模块引用目录,工具-&gt;设置模块引用...

    易语言程序免安装版下载

    修改高级表格支持库,解决在鼠标按下和抬起之间收到时钟周期事件的情况下,无法收到“被单击”事件的BUG。 3. 修改扩展界面支持库三,解决单击卷帘菜单后导致日期框不能弹出下拉窗口的BUG。 4. 修改XP风格支持库...

    cpp_short

    Day01:xshell操作,固定ip,linux命令,用户创建删除Day02:linux目录文件命令,vim操作Day03:linux命令复习,编译工具链,静态库与动态库Day04:C ++命名空间,const,new&delete,引用,强制转换,环境基本...

    RYUTO:龙人

    例如,用于Fedora: sudo dnf install gcc-c++静态版本为Linux发行版提供了libstdc ++和libgcc静态链接,这些发行版无法提供支持&gt; = C11的库。使用在其基本用例中,仅需要提供Ryuto的输出目录和RNA-Seq使用的库类型...

    Android C++高级编程:使用NDK_Onur Cinar, 于红PDF电子书下载 带书签目录 完整版

    11.3 静态运行库与动态运行库 247 11.4 C++异常支持 247 11.5 C++ RTTI支持 248 11.6 C++标准库入门 249 11.6.1 容器 249 11.6.2 迭代器 250 11.6.3 算法 251 11.7 C++运行库的线程安全 251 11.8 C++运行库...

    基于SIP开发软件电话的一些资源(转自YOUTOO)

    将libosip源码包的SRC目录下的Test目录内的C源程序随便拷一个到工程时,直接编译(工程设置里照前文方法在link选项里增加osip2.lib,osipparser2.lib引用我们之前成功编译得到的静态库文件)就可以运行(带参数运行,...

    C-Blog V2.0

    动态日历和完美的文章归档功能,在日历可直接选择特定月份和哪一天的日志归档 .搜索结果采用高亮显示,红字 黄色背景(可按主题和内容搜索) .整合优化了的模版 只需要6个模版文件 .添加模版方便,任何一个plog的...

    LuaBind 源码 (Lua增强库)

    你正在使用一个UNIX系统(或者 cygwin),他们将使得构建LuaBind静态库变得很简单.如果 你正在使用 Visual Studio ,很简单的包含 src 目录下的文件到你的工程即可. 构建LuaBind的时候,你可以设定一些选项来使得库更加...

    Python灰帽子-黑客与逆向工程师的Python编程之道[简体中文版]

    出现在本书中的相当一部分Python代码实例借鉴或直接来源于一些优秀的开源安全项目,诸如Pedram Amini的Paimei,由此读者可以领略到安全研究者们是如何将黑客艺术与工程技术优雅融合来解决那些棘手问题的。...

Global site tag (gtag.js) - Google Analytics