Linux动态库常见问题之-"cannot open shared object file No such file or directory"的解决办法

简介: Linux编写程序时,会经常的接触动态库,而在程序运行时可能会遇到类似于:libxxx.so cannot open shared object file No such file or directory的问题,这个代表什么含义,该怎么解决呢?

含义


这个问题代表的含义是,可执行程序在加载libxxx.so库时,找不到该库。至于找不到的情况分为两种:系统里根本不存在libxxx.so库;libxxx.so库在系统中存在,但是ld找不到,即libxxx.so库的位置没有告知ld。


解决办法


对于第一种情况,需要将libxxx.so放到系统ld的搜索路径中;对于第二种情况,通过locate工具定位libxxx.so的位置,然后将其放到ld的搜索路径中。


动态库的搜索路径配置


Linux下ld对于动态库的搜索路径的配置方式包括以下几种方式:


  1. 通过配置gcc编译器的参数-Wl,-rpath指定;


  1. 通过LD_LIBRARY_PATH环境变量指定;


  1. 通过/etc/ld.so.conf指定,切记修改完ld.so.conf之后必须执行/sbin/ldconfig -v同步动态库;


  1. 默认搜素路径/lib、/usr/lib/指定;


同时,上述几种方式存在一定的搜索顺序,按照搜索的先后依次为:1 > 2 > 3 > 4。对于动态库的搜索路径的配置可以参考21aspnet的这篇博文


依赖动态库的查看


我们可以通过ldd(x86环境)或者CROSS_COMPILE-readelf -d(ARM环境,CROSS_COMPILE为具体交叉编译前缀,-d 为读取动态库信息的参数)来分析相应平台下可执行程序所依赖的动态库的情况,我们可以举一例:


test.c:
#include <stdio.h>                                                                                                                                                                                                  
void test() 
{
    printf("test()\n");
}
$ $(CC) -fPIC -c test.c //生成test.o,-fPIC(Position-independent code)为了生成位置无关的代码,一般用于动态库编译选项。
$ $(CC) -static -fPIC -o libtest.so test.o
extern void test();                                                                                                                                                                                                 
main.c:
int main()
{
    test();
    return 0;
}
$ $(CC) -c main.c
$ $(CC) main.o -o main -L. -ltest
其中,CC代表不同平台的编译器;


通过,修改CC为gcc或者arm-linux-gnueabi-gcc分别生成X86、ARM平台下的main可执行程序,下面分别通过ldd和arm-linux-gnueabi-readelf查看main所依赖的动态库:


  • X86:


ldd main
  linux-vdso.so.1 =>  (0x00007ffc98f6a000)
  libtest.so => /home/qihua/develop/linux/static_dynamic_libs/code/libtest.so (0x00007fe850df4000)
  libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fe850a10000)
  /lib64/ld-linux-x86-64.so.2 (0x000055950a132000)


  • ARM:


arm-linux-gnueabi-readelf -d  main
  Dynamic section at offset 0xf10 contains 25 entries:
    标记        类型                         名称/值
   0x00000001 (NEEDED)                     共享库:[libtest.so]
   0x00000001 (NEEDED)                     共享库:[libc.so.6]
   0x0000000c (INIT)                       0x10430
   0x0000000d (FINI)                       0x10590
   0x00000019 (INIT_ARRAY)                 0x20f08
   0x0000001b (INIT_ARRAYSZ)               4 (bytes)
   0x0000001a (FINI_ARRAY)                 0x20f0c
   0x0000001c (FINI_ARRAYSZ)               4 (bytes)
   0x00000004 (HASH)                       0x101ac
   0x00000005 (STRTAB)                     0x10300
   0x00000006 (SYMTAB)                     0x10200
   0x0000000a (STRSZ)                      200 (bytes)
   0x0000000b (SYMENT)                     16 (bytes)
   0x00000015 (DEBUG)                      0x0
   0x00000003 (PLTGOT)                     0x21000
   0x00000002 (PLTRELSZ)                   32 (bytes)
   0x00000014 (PLTREL)                     REL
   0x00000017 (JMPREL)                     0x10410
   0x00000011 (REL)                        0x10408
   0x00000012 (RELSZ)                      8 (bytes)
   0x00000013 (RELENT)                     8 (bytes)
   0x6ffffffe (VERNEED)                    0x103e8
   0x6fffffff (VERNEEDNUM)                 1
   0x6ffffff0 (VERSYM)                     0x103c8
   0x00000000 (NULL)                       0x0


NOTE


  1. 如果一个可执行程序(常见于第三方库提供的bin)依赖了某些动态库,但可执行程序的CPU架构与本地的CPU架构不同(例如,可执行程序为32位,但本地系统为64位,本地系统安装的动态库也为64位的),那么程序运行时,通过会提示:libxxx.so cannot open shared object file No such file or directory。这是,我们通过locate是可以定位到libxxx.so库,并且libxxx.so处于合适的ld搜索路径之下,这是我们通过修改libxxx.so路径是没有用的。解决问题的方式有这么几种:


1)下载与本地系统CPU架构相匹配的程序;


2)在本地系统重新编译该程序;


3)编译、更新与可执行程序CPU架构相匹配的动态库;


相关文章
|
2月前
|
Linux 开发工具
Linux yum 使用时提示 获取 GPG 密钥失败Couldn‘t open file RPM-GPG-KEY-EPEL-7
Linux yum 使用时提示 获取 GPG 密钥失败Couldn‘t open file RPM-GPG-KEY-EPEL-7
166 3
|
12天前
|
Linux
【Linux系统】使用g = GloveEmbedding()报错BadZipFile: File is not a zip file
本文讨论了在使用embeddings工具包时遇到的“BadZipFile: File is not a zip file”错误,原因是程序中断导致zip文件损坏,解决方法是删除损坏的文件并重新运行程序,具体操作是在Linux系统中删除“~/.embeddings/”目录下的glove文件夹。
23 0
|
2月前
|
Ubuntu Linux 数据安全/隐私保护
Linux yum 使用时提示 获取 GPG 密钥失败Couldn‘t open file RPM-GPG-KEY-EPEL-7
Linux yum 使用时提示 获取 GPG 密钥失败Couldn‘t open file RPM-GPG-KEY-EPEL-7
42 3
|
2月前
|
Linux 开发工具
Linux yum 使用时提示 获取 GPG 密钥失败Couldn‘t open file RPM-GPG-KEY-EPEL-7
Linux yum 使用时提示 获取 GPG 密钥失败Couldn‘t open file RPM-GPG-KEY-EPEL-7
34 2
|
5天前
|
NoSQL 关系型数据库 MySQL
Linux学习记录---(1、基本命令)
该博客文章提供了Linux系统中基本命令的使用记录,包括文件和目录操作、Redis服务管理、MySQL数据库操作以及Tomcat服务器的启动和检查。
Linux学习记录---(1、基本命令)
|
Linux
Linux file 命令详解
哈哈,其实我不是要讲解file命令啦!只是介绍一个小小的技巧,就是如何用file命令查看当前的操作系统位数。 file /bin/任意外部命令 如:file /bin/ls
974 0
|
3天前
|
Linux
会玩这10个Linux命令,一定是个有趣的IT男!
会玩这10个Linux命令,一定是个有趣的IT男!
|
6天前
|
运维 Ubuntu Linux
Linux系统之ncdu命令的基本使用
【8月更文挑战第8天】Linux系统之ncdu命令的基本使用
14 2
Linux系统之ncdu命令的基本使用
|
2天前
|
安全 Linux 开发者
Linux笔记之ldd命令详解
`ldd`命令是Linux环境下一个非常实用的工具,用于显示一个程序运行时所需的共享库依赖。它帮助开发者和系统管理员快速诊断程序运行问题,特别是在处理"找不到库文件"或者"错误的库文件版本"等错误时。然而,出于安全的考虑,对于不信任的可执行文件,应该慎用 `ldd`命令,可以考虑使用其他工具如 `objdump`。总的来说,懂得如何妥善且安全地使用 `ldd`,对于维护一个稳定和高效的Linux系统来说,是非常重要的。
18 9