Linux下动态库(.so)和静态库(.a)

动态库和静态库的介绍

        一般情况下,在项目里会把功能相似的代码封装成库,方便使用和管理,同时增加了代码的内聚性。

库分为两种,一种为静态库,文件名以.a结尾,另一种是动态库,文件名以.so结尾。静态库和动态库的使用各有利弊。

静态库的特点:

  • 简单,链接时直接把静态库中的内容链接到可执行文件中,链接完成后,就不再需要静态库了。
  • 静态库不能共享,所有使用静态库的可执行文件中,都有静态库中的内容,增大了可执行文件的体积,增长了可执行文件装载的时间。。

动态库的特点:

  • 动态库在程序运行时才载入,而且程序可以根据需求自主选择载入动态库的时机,所以使用动态库可以加快可执行文件装载的时间。
  • 动态库是可以共享的,多个程序在运行中可以使用同一个动态库,可以减少程序总的大小。

一个栗子

        现在用一个微型的工程,来讲述静态库、动态库的生成和使用。假设有3个.c文件,分别为:add.c sub.c main.c,实现了add()和sub()这两个函数,功能是返回两个整数相加或者相减的结果,main.c中使用到了这两个函数。下面讲述怎样把这add.c和sub.c两个文件生成并使用静态库和动态库,然后生成可执行文件。

//add.h:
int add(int a, int b);
 
//add.c:
int add(int a, int b) {
    return a + b;
}
 
//sub.h:
int sub(int a, int b);
 
//sub.c:
int sub(int a, int b) {
    return a - b;
}
 
//main.c:
#include "add.h"
#include "sub.h"
#include <stdio.h>
int main() {
    printf("Result of 1 + 1 is: %d\n", add(1,1));
    printf("Result of 6 - 3 is: %d\n", sub(6,3));
 
    return 0;
}

1. 生成静态库

 

生成静态库一般使用 ar 命令,语法为:

ar  archivefile.a objfile.o

# archivefile.a 静态库的名称

# objfile.o是目标文件名,可以多个并列
参数参数意义
-r将objfile文件插入静态库尾或者替换静态库中同名文件
-x 从静态库文件中抽取文件objfile
-t打印静态库的成员文件列表
-d 从静态库中删除文件objfile
-s重置静态库文件索引
-v创建文件冗余信息
-c创建静态库文件

 

通常步骤是把生成静态库所涉及到的.c文件全都编译,生成.o文件,然后使用ar命令生成.a文件,即静态库。这里需要使用到gcc的-c选项,意思是只进行编译,不进行链接。

gcc -W -c add.c -o add.o
gcc -W -c sub.c -o sub.o
ar cvr libaddsub.a add.o sub.o

此时得到了静态库libaddsub.a。可以用命令nm来查看.a中的符号名称。

2. 使用静态库

        使用静态库的方法非常简单,直接把静态库的名字加入到需要编译的文件里就可以。如下所示,注意被依赖的静态库应该放在后面

gcc -W -c main.c -o main.o
gcc main.o libaddsub.a -o test

执行test得到如下结果:

Result of 1 + 1 is: 2
Result of 6 - 3 is: 3

3. 生成动态库
        生成动态库要求编译出的.o文件是位置无关的代码,也就是说需要使用-fpic选项。然后使用gcc的-shared选项生成动态库。

gcc -W -c add.c -fpic -o add.o
gcc -W -c sub.c -fpic -o sub.o
gcc -shared add.o sub.o -o libaddsub.so 

4. 使用动态库
        生成可执行文件的时候需要指定动态库所在的位置,以及动态库的名称。-L选项表示指定动态库所在的目录,-l选项表示需要链接的动态库的名称。一般Linux下动态库的明明规则都是libxxx.so,其中xxx为动态库的真正名称,所以-l选项的写法是-lxxx。-L选项与它后的内容以及-l选项与它后面的内容之间可以没有空格,也可以加上空格。也就是说,-lxxx和-l xxx都是可以的。

gcc -W -c main.c -o main.o
gcc main.o -o test -L./ -laddsub

        如果此时直接运行test的话,会得到类似下面的错误:

./test: error while loading shared libraries: libaddsub.so: cannot open shared object file: No such file or directory

       意思是libaddsub.so不在系统默认加载动态库的目录里面。解决的办法有很多种:

  • (1) 系统默认的动态库目录为/lib*,/usr/lib*,和/etc/ld.so.conf文件中配置的目录。所以可以把libaddsub.so复制到默认的动态库加载目录里,比如/usr/lib目录下。一般安装程序时会这么做,安装完后程序就可以正常启动了。
  • (2) 修改/etc/ld.so.conf配置文件。把所需的动态库所在的目录添加到这个文件里。
  • (3) 启动程序之前向环境变量LD_LIBRARY_PATH中加入动态库所在的目录。可以使用export,或者这样:
   LD_LIBRARY_PATH=$LD_LIBRARY_PATH:PWD ./test

        这只是一个临时的做法,不建议使用这个方法。

  • (4) 生成可执行文件时加入-Wl, -rpath=xxx参数,其中xxx是动态库所在的目录,这里最好写成绝对路径。-Wl表示后面的参数是链接器ld的参数而不是gcc的参数。-rpath表示运行时加载动态库的目录。这样做的话就可以直接运行可执行文件了,因为动态库所在目录的信息已经保存在了可执行文件中。例如:
gcc main.o -o test -L./ -laddsub -Wl,-rpath=./
./test

       运行程序后,还是可以得到和使用静态库相同的结果。

 

 

来源:https://blog.csdn.net/simmerlee/article/details/52758255

参考:https://blog.csdn.net/felixit0120/article/details/7652907

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值