静态库和共享库

库的概念

当进行编辑C代码的时候常常会用到printf函数,这个函数被声明在stdio.h头文件中。这里需要注意的是声明和定义是两个不同的概念:**定义是指为变量、函数或结构体等分配内存空间并初始化的过程。声明:声明是指告诉编译器某个变量、函数或结构体的类型和名称,但不分配内存空间。声明通常只包含类型和名称,不包含初始值。**例如在stdio.h头文件中的printf函数的声明如下:

image-20240830105029085

可以看到在stdio.h中并没有printf函数具体的实现方法,printf函数的实现在/usr/lib目录下的一个后缀名为.so的文件里,也就是说函数的具体实现方法对我们而言是不可见的,头文件仅仅是给我们用户提供一个调用接口供用户使用。而在实际使用中,内核已经帮我们去实现了这两者间的映射关系,所以能够通过直接包含头文件就能够使用它里边的函数。

函数库

  • 函数库是由系统建立的具有一定功能的函数的集合。库中存放函数的名称和对应的目标代码,以及链接过程中所需的重定位信息。(简单来说就是函数的具体实现方法)

  • Linux中标准的C函数库放置在/usr/lib下,通常以后缀名为.so.a文件存放

  • 用户也可以根据自己的需要建立自己的用户函数库(通过对目标文件进行创建)

  • 函数库分为静态库(.a)和共享库(动态库.so

库函数

  • 存放在函数库中的函数,库函数具有明确的功能、入口调用参数和返回值
  • 库函数的源代码一般是不可见的,但在对应的头文件中可以看到对外的接口(函数原型)

一般来说用户都会将源文件编译成目标文件(二进制文件),然后使用工具将目标文件创建静态库或者动态库,这样做的好处一来是可以方便调用,二来是可以保护源代码的知识产权。

静态库的创建与使用

静态库的概念

  • 静态库就是一些.o目标文件的集合,以.a结尾

  • 静态库在程序链接的时候使用,链接器会将程序中使用到函数代码从库文件中拷贝到可执行文件中。一旦链接完成,在执行程序的时候就不需要静态库了。

    有关编译链接的知识可以查看:一个C程序变为一个可执行程序的过程-CSDN博客

  • 由于每个使用静态库的应用程序都需要拷贝所用函数的代码,所以生成的可执行文件会比较大

静态库的创建

#ar rcs lib库文件名.a 目标文件1 目标文件2 .... 目标文件n

#- ar: 是归档工具,用于处理静态库和目标文件。
#- rcs: 是归档命令的选项,其中 `r` 表示替换现有库中的同名文件,`c` 表示创建新的库(如果不存在),`s` 表示写入索引到库中。
#- lib库文件名.a: 这是要创建的静态库的文件名,通常以 `.a` 结尾。
#- 目标文件1 目标文件2 .... 目标文件n: 这些是要添加到静态库中的源文件或目标文件。

代码示例–静态库的创建和使用

使用之前的代码,源文件是io.c,里边包含文件的拷贝,文件权限等函数,目标文件是io.o文件,最后要被cp.c这个文件调用实现文件的拷贝这个功能。之前的编译流程是将源文件io.c编译成目标文件io.o,然后使用gcccp.c、io.o指定头文件路径和输出文件的路径,指令如下:

gcc src/cp.c obj/io.o -o bin/cp -I include/

现在使用静态库的方法进行编译执行

静态库的创建

image-20240830115043783

这里使用ar工具对目标文件创建静态库的时候最好加一个lib前缀,而且后缀名要以.a结尾,这样做的目的是为了后边和源文件的编译运行。

静态库的使用

#方法一:
gcc -o 可执行文件 调用者的目标文件(将源文件编译成目标文件后使用) -Ldir -l库文件名

#方法二:
gcc -o 可执行文件 -Idir 调用者的源文件 -Ldir -l库文件名

如图所示,使用后缀名为.o的目标文件创建成缀名为.a静态库文件,然后将源文件cp.c一起编译。

image-20240830120535056

这里需要使用-L的参数来指定库文件的路径和使用-l的参数指定要链接到哪一个库,链接库的时候不用写lib前缀和.a后缀,直接掐头去尾写中间的名字即可。

  • -Ldir选项表示将指定的库文件所在目录加入到库搜索路径中,默认的库搜索路径在/usr/lib目录下

  • -lname选项(前缀lib不要加上)表示库搜索路径下的libname.alibname.so文件,这也是为什么库文件都以lib开头的原因之一,如果库文件不是以lib开头(例如hello.a),那么就不能用-l选项,而是直接写上hello.a,将原来-lhello换成hello.a即可。所以在进行静态库创建的时候前边加lib前缀算一种规范,方便后边的编译链接操作。

  • -l是链接器选项,必须放到被编译和链接的文件后面

  • 删除静态库文件后不会影响到可执行文件的运行。由于在编译链接的过程中在源文件中用到的静态库中的函数等已经被加入到可执行文件里,所以最后生成的可执行文件和静态库已经没有联系,删掉静态库也并不会影响可执行文件的运行。

    image-20240830153943963

共享库的创建和是使用

共享库的概念

  • 共享库即动态链接库,在Linux中以.so(share object)为后缀,Windows中以.dll为后缀
  • 共享库在程序链接时候并不像静态库那样会拷贝使用函数的代码,而只是做些标记。在程序开始启动(实际上在加载程序时)运行的时候,加载所需的函数。
  • 可执行程序在运行的时候仍然需要共享库的支持
  • 共享库链接出来的文件比静态库要小得多
  • C语言中所有标准的共享库放置在/usr/lib目录中

结合上边几句话的概括:当一个可执行程序运行起来就变成了一个进程,而系统会给每一个进程分配一个虚拟内存。其中大部分是用户空间,当共享库被做了标记以后,此时共享库中被标记的函数就会被加载到用户空间的代码段。所以它这个是只有运行的时候才会加载到内存里,正因为这样它的可执行程序会比静态库编译出来的可执行文件会小得多,并且它的每次执行都依赖于动态库,如果动态库删除,可执行文件就不能运行了。

共享库的创建

gcc -shared -fPIC -o lib库文件名.so 目标文件1...目标文件n

#-shared表示是使用共享库
#-fpic或者-fPIC表明创建产生独立目标代码,具体应用取决于平台

image-20240830163025297

共享库的使用

#方法一:
gcc -o 可执行文件 调用者的目标文件 -Ldir -l库文件名

#方法二:
gcc -o 可执行文件 -Idir 调用者的C源文件 -Ldir -l库文件名

image-20240830163139431

  • 运行可执行文件出错解决方案(库文件找不到)

    image-20240830162358435

    解决方法

    • 将生成的动态库拷贝到/usr/lib目录下去(需要超级用户权限)

      sudo cp libio.so /usr/lib
      

      image-20240830162849139

    • 设置一个临时的环境变量(需要注意的是因为是一个临时的环境变量所以换一个终端或者重启一下就不生效了)

      export LD_LIBRARY_PATH=库文件的目录
      

      image-20240830163320380

  • 删除共享库文件后运行可执行文件出错(提示找不到库文件)

    image-20240830163910334

总结

前边说的静态库的链接是将编译好的静态库在链接的过程将所用到的函数等功能复制到可执行文件里去,而动态库的链接是将编译好的动态库做一些标记。当运行可执行程序时,在加载过程中将其标记的函数等功能加载到虚拟内存的代码段中。所以相对比下来通过静态库链接后的可执行文件会比动态库链接后的可执行文件大得多。由于它们实现的原理不一样,所以可执行文件在脱离静态库后还能运行,而动态库不行。

image-20240830170007038

image-20240830170416395

如图所示,首先使用ar创建一个静态库,然后使用gcc工具进行链接,最后生成cp1的可执行文件,然后使用gcc工具创建一个动态库,然后使用gcc工具进行链接,最后生成cp2的可执行文件。然后使用ls -l对比两个可执行文件可以发现经过静态库链接后的文件大小远大于链接动态库后的文件。并且可以发现静态库删掉以后可执行文件仍然能运行,而动态库不行。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

日落星野

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值