问题背景

  • 在做项目的时候,遇到一个so调用问题,既别人提供了一些so库,其中一个so库包含了给我调用的函数,而这个库里面的函数又调用了其他库的函数,这些所有的库都是linux下编译出来的,而项目则是需要在windows下用Qt交叉编译的。这会导致一些问题,(1)如果想要在Qt工具编译项目的时候使用隐式链接方式,把so库链接进去,会出现找不到别人提供的so库中所依赖的linux系统自带的库。(2)如果使用显式链接,既使用dlopen加载so库,然后再使用dlsym加载函数的方式,那么就会出现找不到so库中函数所调用其他库函数的问题。
  • 由于在交叉编译环境中,目前只能使用显示链接的方式调用别人提供的so库,下面就会用一些测试代码,来展示问题是如何出现的以及如何解决问题。

(一)隐式链接和显示链接的结果

  • 首先我们先输出一个最基础的so库libone.so,具体代码如下:

one.h

extern "C" void One();

one.c

#include "one.h"
#include "stdio.h"
void One()
{printf("One calling\n");
}
  • 然后再写第二个so库libtwo.so,用它调用上面的libone.so中的函数。具体代码如下:

two.h

extern "C" void Two();

two.c

#include "one.h"
#include "two.h"
void Two()
{One();
}

main.c

#include "dlfcn.h"
#include "stdio.h"
#include "two.h"
typedef void (*pTwo)();//TEST宏控制是使用dlopen方式还是直接调用Two方式
#define TEST 0
int main()
{#if TESTprintf("start main\n");pTwo Two = NULL;void *pOpen = NULL;pOpen = dlopen("./libtwo.so", RTLD_LAZY);if (pOpen == NULL){printf("load libtwo.so fail\n");return -1;}Two = (pTwo)dlsym(pOpen, "Two");if (Two == NULL){printf("load function Two fail\n");return -1;}
#endifprintf("start to call Two\n");Two();
#if TESTdlclose(pOpen);pOpen = NULL;
#endifreturn 0;
}

makefile文件

all: libone.so libtwo.so main
main: main.cg++ main.c -ldl -L. -ltwo -lone -o main
libtwo.so: two.cg++ two.c -fPIC -shared -o -libtwo.so
libone.so: one.cg++ one.c -fPIC -shared -o -libone.so
clean:rm ./main ./libtwo.so ./libone.so
  • 上面的Makefile文件中,main程序是链接了libone.so和libtwo.so两个库,但是libtwo.so并没有把libone.so链接上。

运行结果:

  • ldd main

  • ldd libtwo.so

  • nm -D libtwo.so

    可以看到One函数符号没有地址的

  • (1)没有打开TEST宏的结果:

  • (2)打开TEST宏的结果:

  • 此时,问题已经出现了,main把两个需要用到的库在编译后已经链接起来了,所以程序中直接调用Two函数的话,可以看到One是被调用起来了。但是如果使用dlopen去加载libtwo.so库,然后再去加载Two函数的方式时,通过nm -D libtwo.so可以看到,One的函数地址是没有的,所以调用Two时,它再调用One就会报错。

  • libtwo.so和libone.so就相当于别人提供的一些库,libtwo.so中的Two函数就是我们需要使用的函数。那么在项目无法使用第一种方式而只能使用dlopen的方式时,怎么解决该问题呢?可以看到main是可以通过编译链接直接调用Two函数的。那么我们就可以再封装一层,把main看成第三个库,将它封装起来,因为它已经是在Linux环境下能够正常编译和运行的程序了。只需要把main封装成so库,再写一个对外的函数,而这个函数里面直接调用Two,那么就间接的(类似于第一种方式直接的)调用Two函数了。下面就是测试代码和运行结果的展示。

(二)解决显示链接问题的方式

封装出第三个库libthree.so
three.h

extern "C" void Three();

three.c

#include "three.h"
#include "two.h"
void Three()
{Two();
}

然后再修改一下makefile文件:

all: libone.so libtwo.so libthree.so main
main: main.cg++ main.c -ldl -L. -lthree -ltwo -lone -o main
//这个libthree.so就相当于上半部分中的main
libthree.so: three.c libtwo.so libone.sog++ three.c -fPIC -shared -L. -ltwo -lone -o libthree.so
libtwo.so: two.cg++ two.c -fPIC -shared -o -libtwo.so
libone.so: one.cg++ one.c -fPIC -shared -o -libone.so
clean:rm ./main ./libthree.so ./libtwo.so ./libone.so

main.c
main中就把Two改成Three就行:

#include "dlfcn.h"
#include "stdio.h"
#include "three.h"
typedef void (*pThree)();//TEST宏控制是使用dlopen方式还是直接调用Two方式
#define TEST 0
int main()
{#if TESTprintf("start main\n");pThree Three = NULL;void *pOpen = NULL;pOpen = dlopen("./libthree.so", RTLD_LAZY);if (pOpen == NULL){printf("load libthree.so fail\n");return -1;}Three = (pThree)dlsym(pOpen, "Three");if (Three == NULL){printf("load function Three fail\n");return -1;}
#endifprintf("start to call Three\n");Three();
#if TESTdlclose(pOpen);pOpen = NULL;
#endifreturn 0;
}

运行结果:

  • (1)没有打开宏的结果:
  • (2)打开宏的结果:

    总结:从最后的结果可以看到,通过再封装一层的方式,解决了一开始展示的找不到函数符号的问题,使得libthree.so无论是直接的调用Three还是动态加载Three的方式,都能成功得调起了Two函数,直至最终One函数的打印被调用起来。

关于Linux中so显式链接(dlopen)找不到函数符号地址的问题相关推荐

  1. 【转】DLL中类的显式链接

    DLL的显式链接在某些时候比隐式链接具有更大的灵活性.比如,如果在运行时发现DLL无法找到,程序可以显示一个错误信息并能继续运行.当你想为你的程序提供插件服务时,显式链接也很有用处. 显式链接到全局C ...

  2. linux命令wget的url路径太长,Linux中WGET下载https链接及WGET命令的详解

    Linux中WGET下载https链接及WGET命令的详解 使用如下的命令下载https链接: ```bash wget -r -np -nd --accept=gz --no-check-certi ...

  3. 谈谈分布式多智能体中的显式协调机制

    本文csdn博客链接:http://blog.csdn.net/screscent/article/details/78742815 本文qq空间链接:http://user.qzone.qq.com ...

  4. vc++ 显式链接dll

    Visual C++ 显式链接 在显式链接下,应用程序必须进行函数调用以在运行时显式加载 DLL.为显式链接到 DLL,应用程序必须: 调用 LoadLibrary(或相似的函数)以加载 DLL 和获 ...

  5. spring三: 装配bean( 在xml中进行显式配置, 在java中进行显式配置)

    ApplicationContext ac = new AnnotationConfigApplicationContext(SpringConfiguration.class); SpringCon ...

  6. linux程序卸载动态库,Intel平台下linux中ELF文件动态链接的加载、解析及实例分析(二): 函数解析与卸载...

    在 IBM Bluemix 云平台上开发并部署您的下一个应用. 相信读者已经看过了 Intel平台下Linux中ELF文件动态链接的加载.解析及实例分析(一): 加载的内容了,了解了ELF文件被加载的 ...

  7. jQuery 中的显式遍历(explicitly iterate)与隐式遍历(implicit iteration)

    jQuery 中的显式遍历(explicitly iterate)与隐式遍历(implicit iteration) 在 jQuery 中,我们把 $() 叫做核心函数. 如果通过核心函数找到的元素不 ...

  8. linux中软链接和硬链接,Linux中软链接和硬链接的区别

    Linux中软链接和硬链接的区别 链接文件: Linux中包括两种链接:硬链接(Hard Link)和软链接(Soft Link),软链接又称为符号链接(Symbolic link). Inode 文 ...

  9. andriod studio中的显式跳转和隐式跳转

    比如要从A----->B 1.显式跳转 在A的activity中的匿名内部类(这里用匿名内部类)中的写如下代码: Intent intent = new Intent(this, BActivi ...

最新文章

  1. 微信架构 支付架构(下)
  2. Kali Linux重新支持RTL8814au芯片
  3. java中有哪几种注释方式_在 Java 中, 有多种注释方法,其中 __________ 适用于单行注释。...
  4. python连接es_Elasticsearch --- 3. ik中文分词器, python操作es
  5. 连接maven_如何在Eclipse中使用Maven工程连接远程的Spark
  6. 知识图谱论文阅读【十二】【KDD2020】 使用贝叶斯图卷积神经网络推荐精确和多样化项目的框架【看不懂,待续】
  7. 如何避免_小红书如何避免降权!
  8. 设置java路径_关于java路径设置
  9. Photoshop CS2序列号大全 官方免费密钥
  10. IOS 中的Notification 学习
  11. 如何将excel文件联系人转换成vcf文件
  12. c语言词法分析程序实验报告,词法分析器实验报告
  13. C语言里程序编译无误但运行会弹出程序已停止运行是为什么?
  14. 如何通过Spring Boot实施Alexa技能
  15. pytest文档24-fixture的作用范围(scope)
  16. 熊猫入金讲为什么seo没效果
  17. NETDMIS5.0位置度评价案例1
  18. 魅蓝5s即将发布 真假快充15日揭晓答案
  19. 数据治理(四):数据仓库数据质量管理
  20. 认认真真学习STL之string

热门文章

  1. Editors Vulnerability Handbook
  2. [Scala基础]-- 伴生类和伴生对象
  3. C# 将服务器文件批量压缩打包下载至本地两种方法(压缩包缓存到服务器下载、压缩包缓存到内存下载)
  4. WIZNet官方博客开通啦!
  5. 《张成功项目管理记(第2版)》一第一章 结缘项目管理
  6. ThinkPHP3.2.3新特性之:数据库设置
  7. 一张图读懂神州数码集团2020全年业绩
  8. idea debug时提示”Method breakpoints may dramatically slow down debugging“的解决办法
  9. Leetcode动态规划——01背包问题
  10. 联想SR650服务器加CPU,开机提示UPI错误,解决经历。