关于动态库和静态库的问题。

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于动态库和静态库的问题。相关的知识,希望对你有一定的参考价值。

动态库或静态库是编译连接好了的程序,还是只编译好但没有连接的程序?求解...

动态库 和 静态库 可以看成 是 编译好 但 没有 链接好 的 .obj 文件。
当你写程序调用 它们 时,是通过 链接 把它们 链接进来。

当然,动态库实际情况要复杂些。动态库,你编程序调用它们,链接它们 用的是 .lib 文件,真正运行你的程序时 操作系统 还要 再调用 .DLL 文件, 操作系统系统还要检查 代码共享区里是否已有 运行中的代码段,若有则共享,若无,再从DLL 装入。追问

库中的代码也需要调用“其他的函数”,这些其他的函数在库中已经链接好了?
还是在我们的程序要使用静态或动态库时再链接这些“其他的函数“?

追答

看情况,有的链接好了,有的没有。同一库内调用的链接好了。就同你自己写的有几个文件组成的程序一样--各obj之间没链接。

追问

明白~那就是说,如果库中没链接好,那就要在程序链接的时候链接了?

追答

其实,不用程序员操心,你打了编译链接命令后,编译器为你做它需要做的。

参考技术A 静态库就是你直接使用别人的代码,不过省下了编译时间
动态库的不同在于,运行期还省下了系统空间的开销追问

库中的代码也需要调用“其他的函数”,这些其他的函数在库中已经链接好了?
还是在我们的程序要使用静态或动态库时再链接这些“其他的函数“?

追答

链接动作都是在你生成exe时执行的,前期只是完成了编译过程并隐藏了实现细节

Linux 静态库和共享(动态)库的创建与使用详解

Linux 静态库和共享(动态)库

库的介绍

库是二进制文件, 是源代码文件的另一种表现形式, 是加了密的源代码;
是一些功能相近或者是相似的函数的集合体.

使用库有什么好处

  1. 提高代码的可重用性, 而且还可以提高程序的健壮性;
  2. 可以减少开发者的代码开发量, 缩短开发周期.

库制作完成后, 如何给用户使用

  • 头文件—包含了库函数的声明
  • 库文件—包含了库函数的代码实现

注意: 库不能单独使用, 只能作为其他执行程序的一部分完成某些功能, 也就是说只能被其他程序调用才能使用.

库可分静态库(static library)和共享/动态库(shared library)。

首先先介绍静态库(static library)

静态库(static library)

静态库可以认为是一些目标代码的集合, 是在可执行程序运行前就已经加入到执行码中, 成为执行程序的一部分. 按照习惯, 一般以.a做为文件后缀名.
静态库的命名一般分为三个部分:

  • 前缀:lib
  • 库名称:自定义即可, 如test
  • 后缀:.a

所以最终的静态库的名字应该为:libtest.a

静态库的制作

下面以fun1.c ,fun2.chead.h三个文件为例讲述静态库的制作和使用, 其中head.h文件中有函数的声明, fun1.cfun2.c中有函数的实现.

步骤1:将c源文件生成对应的.o文件

gcc -c func1.c func2.c

或者分别生成.o文件:

gcc -c func1.c -o func1.o
gcc -c func2.c -o func2.o

步骤2:使用打包工具ar将准备好的.o文件打包为.a文件
在使用ar工具是时候需要添加参数rcs

  • r更新、c创建、s建立索引
  • 命令:ar rcs 静态库名 .o文件

下图为ar的帮助文档

ar rcs libtest1.a func1.o func2.o

ar工具创建lib过程

静态库的使用

静态库制作完成之后, 需要将.a文件和头文件一定发布给用户.
假设测试文件为main.c, 静态库文件为libtest1.a, 头文件为head.h
用到的参数:

  • L:指定要连接的库的所在目录
  • l:指定链接时需要的静态库, 去掉前缀和后缀
  • I: 指定main.c文件用到的头文件head.h所在的路径
gcc -o main1 main.c -L./ -ltest1 -I./

运行main1,结果如下图:

源代码

func1.c

/*************************************************************************
    > File Name: func1.c
    > Author: 杨永利
    > Mail: 1795018360@qq.com 
    > Created Time: 2021年12月20日 星期一 09时59分47秒
 ************************************************************************/

int add(int a,int b)

    return a+b;


func2.c

/*************************************************************************
    > File Name: func2.c
    > Author: 杨永利
    > Mail: 1795018360@qq.com 
    > Created Time: 2021年12月20日 星期一 10时00分17秒
 ************************************************************************/

 int del(int a,int b)
 
    return a-b;
 

main.c

/*************************************************************************
    > File Name: main.c
    > Author: 杨永利
    > Mail: 1795018360@qq.com 
    > Created Time: 2021年12月20日 星期一 10时01分15秒
 ************************************************************************/

#include <stdio.h>

int add(int a,int b);
int del(int a,int b);

int main(int argc, char* argv[])
    printf("调用库函数add:1+2=%d\\n",add(1,2));
    printf("调用库函数del:4-3=%d\\n",add(4,3));
    return 0;


静态库的优缺点

优点:

  • 函数库最终被打包到应用程序中,实现是函数本地化,寻址方便、速度快。
    (库函数调用效率==自定义函数使用效率)
  • 程序在运行时与函数库再无瓜葛,移植方便。

缺点:

  • 消耗系统资源较大, 每个进程使用静态库都要复制一份, 无端浪费内存。
  • 静态库会给程序的更新、部署和发布带来麻烦。如果静态库libxxx.a更新了,所有使用它的应用程序都需要重新编译、发布给用户(对于玩家来说,可能是一个很小的改动,却导致整个程序重新下载)。

共享库(shared library)/动态库

共享库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入。不同的应用程序如果调用相同的库,那么在内存里只需要有一份该共享库的拷贝,规避了空间浪费问题。

动态库在程序运行时才被载入, 也解决了静态库对程序的更新、部署和发布会带来麻烦。 用户只需要更新动态库即可,增量更新。

按照习惯, 一般以”.so”做为文件后缀名. 共享库的命名一般分为三个部分:

  • 前缀:lib
  • 库名称:自己定义即可, 如test
  • 后缀:.so
    所以最终的静态库的名字应该为:libtest.so

共享库的制作

步骤1:生成目标文件.o, 此时要加编译选项:-fPIC(fpic)

gcc -fpic -c func1.c func2.c

参数:

  • -fpic:创建与地址无关的编译程序(pic, position independent code), 目的就是为了能够在多个应用程序间共享.

步骤2:生成共享库, 此时要加链接器选项: -shared(指定生成动态链接库)

gcc -shared func1.o func2.o -o libtest2.so

共享库的使用

引用动态库编译成可执行文件(跟静态库方式一样):
用到的参数:

  • -L:指定要连接的库的所在目录
  • -l:指定链接时需要的动态库, 去掉前缀和后缀
  • -I: 指定main.c文件用到的头文件head.h所在的路径
gcc main.c -I./ -L./ -ltest2 -o main2

然后运行:./main2,发现竟然报错了,如下图,显示找不到libtest2.so库

分析为什么在执行的时候找不到libtest2.so库?

  • 当系统加载可执行代码时候, 能够知道其所依赖的库的名字, 但是还需要知道所依赖的库的绝对路径。此时就需要系统动态载入器(dynamic linker/loader)。

    ldd命令可以查看可执行文件依赖的库文件, 执行ldd main2, 可以发现libtest2.so找不到.

  • 对于elf格式的可执行程序,是由ld-linux.so*来完成的, 它先后搜索elf文件DT_RPATH段 — 环境变量LD_LIBRARY_PATH — /etc/ld.so.cache文件列表 — /lib/, /usr/lib目录找到库文件后将其载入内存。

    使用file命令可以查看文件的类型: file main2,显示如下图,该可执行文件为,elf格式。

所以说,一旦你定义了一个共享函数库,你还需要安装它。

那么如何安装共享库?

步骤1:拷贝自己制作的共享库到/lib或者/usr/lib

例如上面案例执行下面语句复制到/usr/lib

sudo cp ~/C/sharelib/libtest2.so /usr/lib


然后在终端运行

sudo ldconfig

步骤2:临时设置LD_LIBRARY_PATH:

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径 
export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/home/yang/C/sharelib/libtest2.so

永久设置, 把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径, 设置到~/.bashrc文件中(编辑~/.bashrc,将上面设置语句加到最后一行), 然后在执行下列三种办法之一:

  • 执行. ~/.bashrc使配置文件生效(第一个.后面有一个空格)
  • 执行source ~/.bashrc配置文件生效
  • 退出当前终端, 然后再次登陆也可以使配置文件生效

永久设置,把export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径,设置到/etc/profile文件中(编辑/etc/profile,将上面设置语句加到最后一行)

步骤3:将其添加到 /etc/ld.so.cache文件中
编辑/etc/ld.so.conf文件, 加入库文件所在目录的路径

运行sudo ldconfig -v, 该命令会重建/etc/ld.so.cache文件


解决了库的路径问题之后, 再次ldd命令可以查看可执行文件依赖的库文件, ldd main2,libtest.so就能找到了

再编译运行,运行成功,如下图:

共享库的特点

  • 动态库把对一些库函数的链接载入推迟到程序运行的时期。
  • 可以实现进程之间的资源共享。(因此动态库也称为共享库)
  • 将一些程序升级变得简单。
  • 甚至可以真正做到链接载入完全由程序员在程序代码中控制(显示调用)

比较静态库和动态库的优缺点

静态库

优点:

  1. 执行速度快, 是因为静态库已经编译到可执行文件内部了
  2. 移植方便, 不依赖域其他的库文件

缺点:

  1. 耗费内存, 是由于每一个静态库的可执行程序都会加载一次
  2. 部署更新麻烦, 因为静态库修改以后所有的调用到这个静态库的可执行文件都需要重新编译

动态库

优点:

  1. 节省内存
  2. 部署升级更新方便, 只需替换动态库即可, 然后再重启服务.

缺点:

  1. 加载速度比静态库慢
  2. 移植性差, 需要把所有用到的动态库都移植.

由于由静态库生成的可执行文件是把静态库加载到了其内部, 所以静态库生成的可执行文件一般会比动态库大.

以上是关于关于动态库和静态库的问题。的主要内容,如果未能解决你的问题,请参考以下文章

请教关于linux中静态库与动态库的问题

动态库和静态库的区别

动态库和静态库的区别

动态链接库和静态链接库的区别简述

静态库和动态库的区别

Linux上静态库和动态库的编译和使用