linux静态库、linux动态库制作、使用,动态库报错:error while loading shared libraries: libxxx.so: cannot open shared o

接上一篇:linux_C/C++程序编译-gcc编译器基础使用、编译阶段、编译优化、命令大全(g++适用)

本次来分享linux下C/C++程序的静态库和动态库的制作和使用,不废话,上菜:

1.linux静态库

1.1.linux静态库说明

  linux的静态库是以.a结尾的,而windows的静态库是以.lib结尾的,本次来介绍的是linux的静态库。
1、优点
   ①寻址方便,加载速度快
   ②库被打包到可执行程序中,直接发布可执行程序即可使用
2、缺点
   ①静态库的代码在编译过程中已经被载入可执行程序,因此体积较大。
   ②如果静态函数库改变了,那么你的程序必须重新编译。
3、使用场合
   ①在核心程序上使用,保证速度,可忽视空间
   ②主流应用于80、90年代,现在很少用

1.2.linux静态库命名格式

静态库命名方式:
  libsort.a   #lib开头,sort静态库名,.a结尾

1.3.linux静态库制作

1、gcc/g++命令生成 *.o文件
使用gcc/g++编译器命令:

gcc a.c b.c c.c -c	#生成二进制文件a.o、b.o、c.o

接下来就要用这些.o文件打包成静态库了。

gcc/g++编译器学习

2、ar打包,得到静态库 libmytest.a

ar rcs libmytest.a a.o b.o c.o 	#使用ar工具将需要的.o打包,若是该目录下有不需要的.o文件,就不要将其打包进来,

若是没有不要的.o文件,也可写成命令:
  ar rcs libmytest.a \*.o

(就是一个打包.o文件的过程)

ar 工具不包含在gcc中
参数说明:

r --> 将文件插入静态库中
c --> 创建静态库,不管库是否存在
s --> 写入一个目标文件索引到库中,或者更新一个存在的目标文件索引。

查看库中的符号(.o文件、函数、全局变量等)命令: nm libmytest.a

注意:自己写程序调用别人的静态库时,可执行文件中只打包使用到的别人的静态库中的.o文件,没有使用到的.o是不被打包到自己的程序里的,即生成的二进制文件是与位置有关的,调用的静态库里的.o会被直接打包进自己写的程序可执行文件中(而动态库不会,动态库只是做了一个位置的记录)。

如图:
在这里插入图片描述

1.4.使用静态库

1.4.1.方法1

命令格式:
  gcc + 源文件 + -L 静态库路径 + -l 静态库名 + -I头文件目录 + -o 可执行文件名

参数说明:
-L --> 指定库所在的路径
-l --> 指定库的名字(小写的l)
	去掉前缀 lib
	去掉后缀 .a
	只留下中间部分
-I --> 头文件目录位置(大写的i)

例如:

gcc main.c -L ./ -l mytest -I ./ -o app	#使用静态库编译程序

1.4.2.方法2

命令格式:
  gcc + 源文件 + -I头文件 -o 可执行文件名称 +路径/libxxx.a
例如:

gcc main.c -I ./ -o app ./mylib/libxxx.a	#使用静态库编译程序

注:生成的静态库需要跟对应的头文件同时发布,头文件中存放的是函数接口(函数声明)

2.linux动态库(共享库)

2.1.linux动态库说明

  linux的动态库是以 .so 结尾的,而windows的动态库是以**.dll** 结尾的,现在来介绍的是linux的动态库(也有的人叫共享库)。

1、机制
   动态库的代码是在可执行程序运行时才载入内存的,在编译过程中仅简单的引用,因此代码体积较小。
2、优点

①节省内存(共享)
②易于更新(动态链接)
	停止运行程序,使用新库覆盖旧库(保证新旧库名称一致,接口一致) “接口”,重新启动程序。

3、缺点
   延时绑定,速度略慢
4、使用场合
   对速度要求不是很强烈的地方都应使用动态库

注意:动态库是否加载到内存,取决于程序是否运行。

2.2.linux动态库命名格式

命令格式:
  libmytest.so   #lib开头,动态库名,.so 结尾

2.3.linux动态库制作

  linux动态库制作和使用,会遇到一些问题,不过这些都不难,博主都给出了简易的解决方案,只需要按照博主的步骤,按部就班就可以解决问题。

2.3.1.生成“与位置无关”的目标文件(.o文件)

命令使用:
gcc -fPIC a.c b.c c.c -c
  参数 -fPIC 表示生成与位置无关代码,用的相对地址(生成的.o文件)
  执行完毕后生成一系列的 .o 文件

  调用动态库时,并不会将调用到对应动态库中.o文件打包到自己写的程序可执行文件中,而是对动态库的位置做了记录,是程序起来了再去加载动态库 (静态库则是不一样,会将.o打包到自己的可执行程序文件中)

2.3.2.制作动态库

使用gcc/g++编译器命令:
gcc -shared -o libmytest.so a.o b.o c.o
  参数:-shared 制作动态库
  -o:生成动态库文件的名称
或:gcc -shared -o libmytest.so \*.o   #打包当前路径下的所有.o文件为动态库,注意,若是当前路径下有无关的.o,也会被打包进去,所以使用*.o的时候最好将当前路径下的无关.o移出去

2.3.3.使用动态库

使用gcc/g++编译器命令:
  gcc main.c -L./ -lmytest -I./ -o app
说明:
  -L --> 指定库所在的路径,
  -l --> 去掉前缀 lib,去掉后缀 .so,只留下中间部分
  -I --> 头文件目录位置

2.4.(问题)动态库报错:找不到

2.4.1.执行生成的可执行文件

  在使用动态库以后,生成自己的可执行文件myapp会运行失败。
报错:
error while loading shared libraries: libxxx.so: cannot open shared object file: No such file or directory
原因:动态库找不到。
./myapp --> 运行失败

这时候我们可以去查看myapp依赖的共享库,查看依赖的共享库命令:ldd myapp

如图:
在这里插入图片描述

问题原因:发现.so找不到,是因为没有给动态链接器(ld-linux.so.2)指定好动态库 libmytest.so 的路径。

/ld-linux-x86-64.so.2:动态链接器,按照这种规则去调用我们的动态库。

2.4.2.解决方案–四种方法

(1)临时解决

  使用命令:export LD_LIBRARY_PATH=库路径,将当前目录加入环境变量,但是终端退出了就无效了。(适用于代码测试)

LD_LIBRARY_PATH介绍:
1、作用

①指定查找共享库(动态链接库)时除了默认路径之外的其他路径
②该路径在默认路径之前查找

2、设置方法
用export命令来设置值
例如:

export LD_LIBRARY_PATH=./mylib		#设置mylib文件夹路径为我们动态库所在的路径
echo $LD_LIBRARY_PATH	#打印该环境变量
(2)永久解决

  export LD_LIBRARY_PATH=./mylib写入家目录下.bashrc文件中,这样就可以永久设置了。
  修改了bashrc文件后,需要将终端关掉再次打开,该文件才能生效。

(3) 永久解决(正规写法)

步骤:

1. 需要找动态连接器的配置文件 --  /etc/ld.so.conf
2. 动态库的路径写到配置文件中 -- 绝对路径,例如:/home/myproject/mylib
3. 终端使用命令更新:sudo ldconfig -v

以上就是本次的分享了,希望能对广大网友有帮助。

此博主在CSDN发布的文章目录:【我的CSDN目录,作为博主在CSDN上发布的文章类型导读

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

futureCode.

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

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

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

打赏作者

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

抵扣说明:

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

余额充值