LINUX 静态库和动态库


       库有动态与静态两种,动态通常用.so为后缀,静态用.a为后缀。例如:libhello.so libhello.a


1,生成库


      第一步要把源代码编绎成目标代码。以下面的代码为例,生成上面用到的hello库:

        /* hello.c */
       #include
       void sayhello()
      {
             printf("hello,world/n");
      }
     用gcc编绎该文件,在编绎时可以使用任何全法的编绎参数,例如-g加入调试代码等:
             gcc -c hello.c -o hello.o
1).连接成静态库
         现在我们创建libhello静态库文件:
         $ gcc -c libhello -o libhello.o
         $ ar rcs libhello.a libhello.o  

         其中ar中的rcs的意思是: r表明将模块 加入到静态库中,c表示创建静态库,s表示生产索引。
2).连接成动态库

 

         gcc    -fpic/fPIC -c source.c -o source.o
            gcc -shared -Wl,-soname,your_soname -o library_name file_list library_list

           说明: 

               -fpic或者-fPIC表明创建position independent code,这通常是创建共享库必须的。
               -Wl 表明给链接器传送参数,所以这里-soname, library_name 为给链接器的参数。
               -shared 表明是使用共享库
          下面是使用a.c和b.c创建共享库的示例:
           gcc -fPIC -g -c -Wall a.c
           gcc -fPIC -g -c -Wall b.c
           gcc -shared -Wl,-soname, libmyab.so.1 -o libmyab.so.1.0.1 a.o b.o -lc

               说明: lc == libc
        生成动态库用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 是次版本号



2,使用库


1) 使用静态库


       当要使用静态的程序库时,连接器会找出程序所需的函数,然后将它们拷贝到执行文件,由于这种拷贝是完整

的,所以一旦连接成功,静态程序库也就不再需要了。 然而,对动态库而言,就不是这样。动态库会在执行程序内

留下一个标记‘指明当程序执行时,首先必须载入这个库。由于动态库节省空间,linux下进行连接 的缺省操作是首

先连接动态库,也就是说,如果同时存在静态和动态库,不特别指定的话,将与动态库相连接。


2) 与动态库连接


      linux默认的就是与动态库连接,下面这段程序testlib.c使用hello库中的sayhello()函数
                   /*testlib.c*/
                   #include
                   #include
                   int main()
                   {
                        sayhello();
                        return 0;
                    }

      使用如下命令进行编译
                   $gcc -c testlib.c -o testlib.o
      用如下命令连接:
                   $gcc testlib.o -lhello -o testlib      

      在连接时要注意,假设libhello.o 和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


3.动态库的路径问题


       为了让执行程序顺利找到动态库,有三种方法:
       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刷新。这样,加入的目录下的所有

库文件都可见。


4.查看库中的符号


      有时候可能需要查看一个库中到底有哪些函数,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



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值