1.动态库与静态库简介

在实际的软件开发中,为了方便使用一些被重复调用的公共代码,我们经常将这些公共的函数编译成动态库或静态库。我们知道程序一般要经过预处理、编译、汇编和链接这几个步骤才能变成可执行的文件,程序的静态库是在做链接的步骤时,通过连接器将静态库的代码copy到可执行文件中,目标文件包含所有的静态库代码。因此目标文件在执行的时候不需要静态库的支持,但是在当静态库改变时则需要重启编译可执行文件。在Linux中,一般静态库是以 XXX.a命名。而动态库不会被编译到目标代码中,而是在执行到库的相关函数时才会去调用。因此当动态库函数改变时,只要保证动态库所存放的路径不变,我们将新的动态库覆盖旧的动态库就OK了, 在Linux中,动态库一般以XXX.so命名。

2.动态库与静态库的实现

2.1 静态库在可以通过gcc编译源代码文件而得到,在Linux中我们可以通过file xxx.a查看其文件属性,用ar -t xxx.a 查看这个静态库包含的目标,如下所示:

 /lib64> file libapparmor.a
libapparmor.a: current ar archive /lib64> ar -t libapparmor.a
grammar.o
libaalogparse.o
kernel_interface.o
scanner.o

2.2 静态库目前在Linux中很少见了,更多用的是动态库, 动态库可以通过 “gcc -shared -fPIC XXX.so xxx.o”编译而成。

-shared该选项指定生成动态连接库(让连接器生成T类型的导出符号表,有时候也生成弱连接W类型的导出符号),不用该标志外部程序无法连接。相当于一个可执行文件。

-fPIC:表示编译为位置独立的代码,不用此选项的话编译后的代码是位置相关的所以动态载入时是通过代码拷贝的方式来满足不同进程的需要,而不能达到真正代码段共享的目的。

我们可以用两个方式来查看该目标文件所引用的动态库(ldd & readelf):

ELF 文件包含了共享库文件信息,我们可以注意到ldd列出来的比readelf列出来的库文件多,这是因为ldd列出来的是实际运行加载的库文件, 而readelf列出来的是编译时链接的库文件。

 ldd /usr/bin/ls
linux-vdso.so. (0x00007ffe655c0000)
libselinux.so. => /lib64/libselinux.so. (0x00007fdeeee47000)
libcap.so. => /lib64/libcap.so. (0x00007fdeeec42000)
libc.so. => /lib64/libc.so. (0x00007fdeee89e000)
libpcre.so. => /usr/lib64/libpcre.so. (0x00007fdeee62f000)
libdl.so. => /lib64/libdl.so. (0x00007fdeee42b000)
/lib64/ld-linux-x86-.so. (0x0000556694a1a000)
libpthread.so. => /lib64/libpthread.so. (0x00007fdeee20d000) readelf -d /usr/bin/ls Dynamic section at offset 0x1cd98 contains entries:
Tag Type Name/Value
0x0000000000000001 (NEEDED) Shared library: [libselinux.so.]
0x0000000000000001 (NEEDED) Shared library: [libcap.so.]
0x0000000000000001 (NEEDED) Shared library: [libc.so.]
0x000000000000000c (INIT) 0x402770
0x000000000000000d (FINI) 0x4141ac
0x0000000000000019 (INIT_ARRAY) 0x61c368
0x000000000000001b (INIT_ARRAYSZ) (bytes)
0x000000000000001a (FINI_ARRAY) 0x61c370
0x000000000000001c (FINI_ARRAYSZ) (bytes)
0x0000000000000004 (HASH) 0x400298
0x000000006ffffef5 (GNU_HASH) 0x4006e0
0x0000000000000005 (STRTAB) 0x401468
0x0000000000000006 (SYMTAB) 0x400748
0x000000000000000a (STRSZ) (bytes)
0x000000000000000b (SYMENT) (bytes)
0x0000000000000015 (DEBUG) 0x0
0x0000000000000003 (PLTGOT) 0x61d000
0x0000000000000002 (PLTRELSZ) (bytes)
0x0000000000000014 (PLTREL) RELA
0x0000000000000017 (JMPREL) 0x401cf0
0x0000000000000007 (RELA) 0x401c00
0x0000000000000008 (RELASZ) (bytes)
0x0000000000000009 (RELAENT) (bytes)
0x000000006ffffffe (VERNEED) 0x401b90
0x000000006fffffff (VERNEEDNUM)
0x000000006ffffff0 (VERSYM) 0x401a72
0x0000000000000000 (NULL) 0x0

动态库的加载有两个,一种是按照库的路径搜索并加载,加载的优先级如下:

  1、RUNPATH 优先级最高    
  2、RPATH   其次               编译时指定
  3、LD_LIBRARY_PATH    环境变量指定
  4、/etc/ld.so.cache            可以通过ldconfig修改
  5、/usr/lib/ /lib/                  

另一种加载方式是依赖于Linux提供的API 来完成, API提供的函数声明,包含头文件:dlfcn.h.

关于动态库和静态库就说这么多了,水平有限,错误请指正,谢谢!

Linux的动态库与静态库的更多相关文章

  1. Linux中的动态库和静态库(.a/.la/.so/.o)

    Linux中的动态库和静态库(.a/.la/.so/.o) Linux中的动态库和静态库(.a/.la/.so/.o) C/C++程序编译的过程 .o文件(目标文件) 创建atoi.o 使用atoi. ...

  2. Linux系统中“动态库”和“静态库”那点事儿 /etc/ld.so.conf 动态库的后缀为*.so 静态库的后缀为 libxxx.a ldconfig 目录名

    Linux系统中“动态库”和“静态库”那点事儿 /etc/ld.so.conf  动态库的后缀为*.so  静态库的后缀为 libxxx.a   ldconfig   目录名 转载自:http://b ...

  3. Linux系统中“动态库”和“静态库”那点事儿【转】

    转自:http://blog.chinaunix.net/uid-23069658-id-3142046.html 今天我们主要来说说Linux系统下基于动态库(.so)和静态(.a)的程序那些猫腻. ...

  4. 技巧:Linux 动态库与静态库制作及使用详解

    技巧:Linux 动态库与静态库制作及使用详解 标准库的三种连接方式及静态库制作与使用方法 Linux 应用开发通常要考虑三个问题,即:1)在 Linux 应用程序开发过程中遇到过标准库链接在不同 L ...

  5. Linux系统中“动态库”和“静态库”那点事儿

    摘自http://blog.chinaunix.net/uid-23069658-id-3142046.html 今天我们主要来说说Linux系统下基于动态库(.so)和静态(.a)的程序那些猫腻.在 ...

  6. linux动态库与静态库混合连接

      1, 在应用程序需要连接外部库的情况下,linux默认对库的连接是使用动态库,在找不到动态库的情况下再选择静态库.使用方式为: gcc test.cpp -L. -ltestlib 如果当前目录有 ...

  7. Linux环境编译动态库和静态库总结

    对Linux环境动态库和静态库的一些基础知识做一些总结, 首先总结静态库的编译步骤. 1 先基于.cpp或者.c文件生成对应的.o文件 2将几个.o文件 使用ar -cr命令 生成libname.a文 ...

  8. Linux下编译、使用静态库和动态库 自己测过的

    每个程序实质上都会链接到一个或者多个的库.比如使用C函数的程序会链接到C运行时库,GUI程序会链接到窗口库等等.无论哪种情况,你都会要决定是链接到静态库(static libary)还是动态库(dyn ...

  9. 自己在linux上编译、链接、动态库和静态库的学习笔记

    在平常的项目中,我们都是使用公司要求的makefile.makedebug一类的文件,因此,在编译.链接.生成和链接动态库与静态库的时候,我们只是简单的使用一些已经设置的变量,只是简单的修改.添加一些 ...

随机推荐

  1. Android进阶(十)Android 发邮件

    最近在做的APP涉及到发邮件,总结如下: 在android里进行邮件客户端开发可以有两种方式: 在邮件客户端的设计中,可以采用两种方法. 一种是调用android系统自带的邮件服务 优点:这种方法比较 ...

  2. mixer: mysql协议分析

    综述 要实现一个mysql proxy,首先需要做的就是理解并实现mysql通讯协议.这样才能通过proxy架起client到server之间的桥梁. mixer的mysql协议实现主要参考mysql ...

  3. android TabLayout实现京东详情效果

    Google在2015的IO大会上,给我们带来了更加详细的Material Design设计规范,同时,也给我们带来了全新的Android Design Support Library,在这个supp ...

  4. java,http的post和get

    使用Java发送GET.POST请求 --节选自<疯狂Java讲义>    URL的openConnection()方法将返回一个URLConnection对象,该对象表示应用程序和 UR ...

  5. 如何实现 集群化/Session 复制-doc(cluster-howto.html)

    源文档链接: http://tomcat.apache.org/tomcat-6.0-doc/cluster-howto.html 翻译日期: 2014年3月19日 翻译人员: 铁锚 感受: Tomc ...

  6. Android下NDK开发环境搭建

    Android下NDK开发环境搭建 1.     AndroidNDK安装与配置 1.1  NDK简介 Android NDK是一套允许开发人员使用本地代码(如C/C++)进行Android APP部 ...

  7. 高斯混合模型和EM算法

    使用期望最大化算法(Expectation-Maximization)来进行密度估计(density estimation). 与k-means一样,给定的训练样本是,我们将隐含类别标签用表示.与k- ...

  8. RedHat系列软件管理(第二版) --源码包安装

    RedHat系列软件管理 --源码包安装 源码包特点: 拥有广泛的平台支持性,可以装在所有的类UNIX操作系统上,不用考虑CPU架构. 灵活性,可以在安装过程中指定特有的选项. 定制度非常高,可以自己 ...

  9. ubuntu下如何安装codeblocks集成开发环境

    codeblocks是一个十分优秀的C/C++开发IDE,虽然后起之秀codelite目前来看大有超越之势哦. 不过在ubuntu下安装codeblocks却比较麻烦,不像其他linux发行版,比如s ...

  10. python select.select模块通信全过程详解

    要理解select.select模块其实主要就是要理解它的参数, 以及其三个返回值.select()方法接收并监控3个通信列表, 第一个是所有的输入的data,就是指外部发过来的数据,第2个是监控和接 ...