Linux下库相关的概略介绍及动态库的搜索路径优先级

(本文主要针对动态连接库,内容来源网络,重新整理之,高亮部分的请重点关注)

 

一. 基本知识
  总 体来说, 库可以有三种使用的形式: 静态、共享和动态. 静态库的代码在编译时就已连接到开发人员开发的应用程序中, 而共享库只是在程序开始运行时才载入. 在编译时, 只是简单地指定需要使用的库函数. 动态库则是共享库的另一种变化形式. 动态库也是在程序运行时载入, 但与共享库不同的是, 使用的库函数不是在程序运行开始, 而是在程序中的语句需要使用该函数时才载入. 动态库可以在程序运行期间释放动态库所占用的内存, 腾出空间供其它程序使用. 由于共享库和动态库并没有在程序中包括库函数的内容, 只是包含了对库函数的引用,因此代码的规模比较小.

  而Linux下的库文件分为共享库和静态库两大类, 它们两者的差别仅在程序执行时所需的代码是在运行时动态加载的, 还是在编译时静态加载的. 区分库类型最好的方法是看它们的文件后缀,通常共享库以.so(Shared Object的缩写)结尾, 静态链接库通常以.a结尾(Archive的缩写). 在终端缺省情况下, 共享库通常为绿色, 而静态库为黑色.

  已经开发的大多数库都采取共享库的方式. ELF格式的可执行文件使得共享库能够比较容易地实现, 当然使用旧的a.out模式也可以实现库的共享. Linux系统中目前可执行文件的标准格式为ELF(Executable and Linkable Format) 格式(可点击查看此定义).

  .a的是为了支持较老的a.out格式的可执行文件.
  .so的是支持elf格式的可执行文件的库.

    .a是静态库文件, 可以用ar 命令生成.
  .so是动态库文件, 编译时加上指定的选项即可生成.

 

二.命名规则
      GNU库的使用必须遵守Library GNU Public License(LGPL许可协议). 该协议与GNU许可协议略有不同, 开发人员可以免费使用GNU库进行软件开发, 但必须保证向用户提供所用的库函数的源代码.

  系统中可用的库都存放在/usr/lib和/lib目录中. 库文件名由前缀lib和库名以及后缀组成. 根据库的类型不同, 后缀名也不一样.
  共享库的后缀名由.so和版本号组成, 静态库的后缀名为.a.

   libname.so.major.minor.patchlevel

   name : 可以是任何字符串, 用来唯一标识某个库. 该字符串可以是一个单字、几个字符、甚至一个字母.
   major : 主版本号
   minor : 次版本号

   patchlevel : 补丁版本

 

三.库操作 (具体操作本文不详述)
   ldd( Library Dependency Display ) : 显示一个executable program对库的依赖关系 .
 

   ldconfig - configure dynamic linker run-time bindings : 他的作用就是为了让动态链接库被系统所认识及共享 .
   默认搜寻/lib /usr/lib 以及动态库配置文件/etc/ld.so.conf 内所列的目录, 创建出动态装入程序(ld.so )所需的连接和缓存文件 (/etc/ld.so.cache )
—————————————————————————————————————
   关于 ld.so 此处多说几句
   共享库装载器& 动态连接器 : 当程序被调用的时候, Linux 共享库装载器也自动被调用. By the way , ‘ 共享库装载器’ 这个名词的定义不知正确否. 但无论是什么 , 它的作用是保证程序所需要的所有适当版本的库都被调入内存. 而可能的共享库装载器的名字就是 ld.so 或者是 ld-linux.so (取决于 Linux libc 的版本)
—————————————————————————————————————

四.动态库的搜索路径优先级

   Section 3 提到*.so被寻在/lib /usr/lib 以及/etc/ld.so.conf 内所列的目录中 , 这就涉及到优先搜索哪个路径文件夹的问题 . 于是 , 解渴先 :
   动态链接器ld.so按照下面的顺序来搜索需要的动态共享库:

   1.ELF可执行文件中动态段中DT_RPATH所指定的路径. 这实际上是通过一种不算很常用, 却比较实用的方法所设置的: 编译目标代码时, 可以对gcc加入链接参数“-Wl,-rpath”指定动态库搜索路径;
   2.环境变量LD_LIBRARY_PATH 指定的动态库搜索路径;
   3./etc/ld.so.cache中所缓存的动态库路径(如果支持ld.so.cache的话) . 这可以通过修改配置文件/etc/ld.so.conf中指定的动态库搜索路径来改变;
   4.默认的动态库搜索路径/lib;
   5.默认的动态库搜索路径/usr/lib.
    
   Comment : 在嵌入式Linux系统的实际应用中, 1和2被经常使用, 也有一些相对简单的的嵌入式系统会采用4或5的路径来规范动态库. 3在嵌入式系统中使用的比较少, 因为有很多系统根本就不支持ld.so.cache.

   if(!you care about how i get this result){
         echo The End . BYEBYE!
   }
   else
         # gcc -o pos main.c -L. -lenvpos

         #

               void pos();
                   int main()
                   {
                        pos();
                        return 0;
                   }
         然后在上述5种或者更多的路径下,分别编译出libenvpos.o(注意要同名啊).

—————————————————————————————————————
        # gcc -c pos_env.c
                      # gcc -shared -fPCI -o libenvpos.so pos_env.o
                      #

                      #include
                        void pos()
                        {
                               printf(”/yourURL/lib\n”);
                       }
                        program: pos_env.c
—————————————————————————————————————
          执行./pos 看输出结果 , 输出一个结果后就到结果指向的目录下删除*.so之 . 一次这样顺序的搞定也就知道搜索的优先级了 .

          好了 , 就到这里吧 . 各位师兄们有嘛好方法和相关知识点 , 请麻烦给我留言 .

          也在此处发布: http://www.emsym.com/blog/?p=631
—————————————————————————————————————

     Click here to read more about this Topic (en)    &  Click here to download some papers about Dynamic Link Mechanism in Linux(zh-cn)

你可能感兴趣的:(linux,gcc,网络协议,网络应用,嵌入式)