C/C++混合项目,程序运行报错:未定义函数符号

参考

C/C++项目混合编译

extern "C" 详解

《C++ Primer Plus》函数重载篇章

环境

系统:ubuntu20

编译器:gcc

问题

C 和 C++ 源码的混合项目,编译成功,但是程序运行报错找不到函数符号。

背景

项目中使用第三方的代码,是 C 文件,但项目本身是 C++ 工程。

CMakeList.txt 语法检查没有问题,在工程的 devel/lib 文件夹能够找到生成的 so 库。

so 库符号检查

检查报错的 local_planner.cpp 生成的 liblocal_planner.so 文件。查看库有哪些符号丢失。

ldd -r liblocal_planner.so
...undefined symbol: _Z18dubins_path_lengthP10DubinsPath        (./liblocal_planner.so)
undefined symbol: _Z20dubins_shortest_pathP10DubinsPathPdS1_d        (./liblocal_planner.so)
undefined symbol: _Z18dubins_path_sampleP10DubinsPathdPd        (./liblocal_planner.so)

上面的三个未找到的函数符号中存在第三方库的函数名:dubins_path_length,dubins_shortest_path,dubins_path_sample。

查看第三方代码生成的 libdubins_lib 库中有哪些符号(函数,变量)

nm -D libdubins_lib.so
                 U acosU atan2U cosw __cxa_finalize
0000000000003000 R DIRDATA
0000000000001e01 T dubins_extract_subpath
0000000000001f57 T dubins_intermediate_results
00000000000028c0 T dubins_LRL
000000000000213f T dubins_LSL
00000000000023d5 T dubins_LSR
00000000000015f4 T dubins_path
0000000000001dc0 T dubins_path_endpoint
0000000000001746 T dubins_path_length
0000000000001a40 T dubins_path_sample
0000000000001cf6 T dubins_path_sample_many
0000000000001841 T dubins_path_type
00000000000026ec T dubins_RLR
000000000000256f T dubins_RSL
0000000000002288 T dubins_RSR
0000000000001856 T dubins_segment
00000000000017be T dubins_segment_length
0000000000001806 T dubins_segment_length_normalized
00000000000013ff T dubins_shortest_path
0000000000002a9e T dubins_wordU floorU fmin
0000000000001399 T fmodrw __gmon_start__w _ITM_deregisterTMCloneTablew _ITM_registerTMCloneTable
00000000000013d2 T mod2piU sinU sqrtU __stack_chk_fail

第三方库 so 文件存在函数符号:dubins_path_length,dubins_shortest_path,dubins_path_sample。且函数符号与函数名一致。

经过对比可以发现,第三方库生成的 libdubins_lib.so 文件中函数符号确实和使用第三方函数的库 liblocal_planner.so 中的函数符号不相同。这也就是为什么程序运行中会找不到函数符号的直接原因。

extern "C"

怀疑是 C 和 C++ 项目混合编译存在问题。搜索后了解到在 C++ 项目中使用 C 源码,是需要在第三方代码的 C 头文件中增加 extern "C" 宏指令。

#ifdef __cplusplus
extern "C" {
#endif// c 头文件内容
...#ifdef __cplusplus
}
#endif

补充宏指令后编译,检查 so 库已经没有那三个函数符号未定义的提示了!

C/C++ 的函数符号规则

当编译的程序是 C++ 工程时候,会定义 __cplusplus 宏,提供给编译器识别。(很多编译器,系统环境变量名都是以两个下划线开头 “__XXX”,所以尽量不要定义两个下划线开头的变量,避免覆盖编译器,环境变量。)

extern 是 C/C++ 语言中表明函数和全局变量作用范围(可见性)的关键字,该关键字告诉编译器,其声明的函数和变量可以在本模块或其它模块中使用。
被 extern "C" 修饰的变量和函数是按照 C 语言方式编译和连接的。

第三方库的函数原型如下:

double dubins_path_length(DubinsPath* path);
int dubins_path_sample(DubinsPath* path, double t, double q[3]);
int dubins_shortest_path(DubinsPath* path, double q0[3], double q1[3], double rho);

由于在 C 中,全局函数名唯一,预处理后的函数符号直接使用函数名即可。所以在其 so 库中,这三个函数符号就是函数名。

0000000000001746 T dubins_path_length
0000000000001a40 T dubins_path_sample
00000000000013ff T dubins_shortest_path

而在未添加 extern "C" 之前,在 C++ 工程中,默认这三个函数都是 C++ 的函数,它们会被处理为 C++ 风格的函数符号。

_Z18dubins_path_lengthP10DubinsPath
_Z20dubins_shortest_pathP10DubinsPathPdS1_d
_Z18dubins_path_sampleP10DubinsPathdPd

在 C++ 中,由于函数可以重载,所以在编译阶段,函数会被编译为 返回值+函数名+参数列表 的唯一符号。其中参数列表称为函数特征标。函数名+函数特征标构成 C++ 函数重载的关键。
不难推测,该编译器将返回值 int 标识为 _Z18,把返回值 double 标识为 _Z20;P10 用来隔离函数名和参数列表。

题外话

为什么 C++ 函数重载不考虑返回值?

由于 C/C++ 的函数返回值并非必须使用的。如果函数重载也考虑了返回值,那么在只有返回值不同的重载函数使用中,编译器会无法确定究竟要使用的是哪个函数。

e.g. 演示了 C++ 函数重载考虑返回值的问题。

double add (int, int); // 函数原型 0
int add (int, int); // 函数原型 1int main() {int a = 10, b =20;add(a, b); // 这里使用的是哪个函数?
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/790561.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

03.31_111期_C++_list特性和使用

利用想定义一个带头双向链表的迭代器 只能通过在 带头双向链表的类 中同一个命名空间中 再定义一个类 这个类规定 带头双向链表中迭代器的行为 下面的设计需要注意: 1. 定义的这个类需要写出构造函数, 1.1 构造函数…

边缘计算盒子与云计算:谁更适合您的业务需求?

边缘计算盒子和云计算,这两个概念听起来可能有点复杂,但其实它们就是两种不同的数据处理方式。那谁更适合您的业务需求呢?咱们来详细说说。 边缘计算盒子,就像是个小型的数据处理中心,放在离你业务现场比较近的地方。它…

Tensorflow2.0笔记 - 自定义Layer和Model实现CIFAR10数据集的训练

本笔记记录使用自定义Layer和Model来做CIFAR10数据集的训练。 CIFAR10数据集下载: https://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz 自定义的Layer和Model实现较为简单,参数量较少,并且没有卷积层和dropout等,最终准确率…

基于java+SpringBoot+Vue的图书个性化推荐系统的设计与实现

基于javaSpringBootVue的图书个性化推荐系统的设计与实现 开发语言: Java 数据库: MySQL技术: SpringBoot MyBatis Vue工具: IDEA/Eclipse、Navicat、Maven 系统展示 前台展示 首页:展示图书信息、好书推荐、留言反馈等。 图书信息:用户可以查看图…

easyExcel 模版导出 中间数据纵向延伸,并且对指定列进行合并

想要达到的效果 引入maven引用 <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.2.1</version></dependency> 按照要求创建模版 备注 : 模板注意 用{} 来表示你要用的变量 如果本…

商务电子邮件: 在WorkPlace中高效且安全

高效和安全的沟通是任何组织成功的核心。在我们关于电子邮件类型的系列文章的第二期中&#xff0c;我们将重点关注商业电子邮件在促进无缝交互中的关键作用。当你身处重要的工作场环境时&#xff0c;本系列的每篇文章都提供了电子邮件的不同维度的视角。 “2024年&#xff0c;全…

01 使用ArcGIS生成节点路径

目录 1 测试数据准备 1.1 创建空的GDB文件及数据集 1.2 创建道路图层 1.3 绘制路网

计算机视觉之三维重建(6)---多视图几何(上)

文章目录 一、运动恢复结构问题&#xff08;SfM&#xff09;二、欧式结构恢复2.1 概述2.2 求解2.3 欧式结构恢复歧义 三、仿射结构恢复3.1 概述3.2 因式分解法3.3 总结3.4 仿射结构恢复歧义 一、运动恢复结构问题&#xff08;SfM&#xff09; 1. 运动恢复结构问题&#xff1a;通…

enqueue:oracle锁机制

实现锁的方式就是排队咯&#xff0c;那么排队就是有enqueue这么个结构来管理 管理锁的结构叫队列&#xff0c;即enqueue 所有和enqueue相关的函数都叫KSQ-- kernal service enqueue lock是从应用层面看到的锁&#xff0c;enqueue是oracle内部管理锁的一个结构。 可以用v$lock_…

python将visio转换为 PDF 文件

参考链接&#xff1a;在 Python 中將 Visio 轉換為 PDF | Python Visio 到 PDF 庫 (aspose.com) 下载软件包&#xff1a; pip install aspose-diagram-python 读取文件&#xff0c;保存为PDF # 此代碼示例演示如何使用 PDF 保存選項將 Visio 轉換為 PDF import aspose.dia…

基于单片机的超声波测距仪设计_kaic

摘 要 如今社会持续深化转型&#xff0c;在人工智能领域&#xff0c;传感器采集外部数据&#xff0c;经过处理器对数 据运算和处理&#xff0c;从而实现相应的功能。比如自动驾驶技术中&#xff0c;超声波传感器应用广泛&#xff0c; 超声波是一种频率在 20khz 以上的声波&…

HTML优化SEO的实用技巧

在网站开发中&#xff0c;除了关注设计和用户体验&#xff0c;SEO&#xff08;搜索引擎优化&#xff09;也是提升网站流量和可见度的关键。合理的HTML结构和元素运用能够帮助搜索引擎更好地理解页面内容&#xff0c;从而提高搜索排名。以下是一些基于HTML的SEO优化技巧&#xf…

OpenHarmony实战:小型系统移植概述

驱动主要包含两部分&#xff0c;平台驱动和器件驱动。平台驱动主要包括通常在SOC内的GPIO、I2C、SPI等&#xff1b;器件驱动则主要包含通常在SOC外的器件&#xff0c;如 LCD、TP、WLAN等 图1 OpenHarmony 驱动分类 HDF驱动被设计为可以跨OS使用的驱动程序&#xff0c;HDF驱动框…

【WebKit架构讲解】

&#x1f308;个人主页:程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

Nginx从安装到高可用实用教程!

一、Nginx安装 1、去官网http://nginx.org/下载对应的nginx包&#xff0c;推荐使用稳定版本 2、上传nginx到linux系统 3、安装依赖环境 (1)安装gcc环境 yum install gcc-c(2)安装PCRE库&#xff0c;用于解析正则表达式 yum install -y pcre pcre-devel(3)zlib压缩和解压缩…

java面试题(3)|解释 null 和 “null“ 之间的区别,并举例说明它们在编程中的使用场景

null 和 "null" 之间的区别主要在于语义和数据类型上&#xff1a; null 是一个特殊的值&#xff0c;通常用于表示缺少有效值或未定义的变量。在许多编程语言中&#xff0c;null是一个关键字&#xff0c;表示空值。例如&#xff0c;在Java中&#xff0c;当一个对象尚…

解决el-table设置固定高度后,展示不同列时表格高度变小bug

解决el-table设置固定高度后&#xff0c;展示不同列时表格高度变小bug 1、需求分析2、解决方案 1、需求分析 在el-table使用过程中&#xff0c;选择多个参数展示更多列时会出现高度变小问题究其原因可知是el-table列动态发生变化后&#xff0c;el-table__body-wrapper的高度变…

sqlite在非主键创建一个自增字段

sqlite 自增比较奇葩&#xff0c;自增字段必须建在主键上&#xff0c;但主键很重要。不是每种情况都是给自增去做。比如要实现replace into 时&#xff0c; 要主键作为更新标识。用自增很难实现。 开工&#xff1a; 1、建立一个主表&#xff0c;主表的ID是自增ID&#xff0c;…

CNAS软件测试公司有什么好处?如何选择靠谱的软件测试公司?

CNAS认可是中国合格评定国家认可委员会的英文缩写&#xff0c;由国家认证认可监督管理委员会批准设立并授权的国家认可机构&#xff0c;统一负责对认证机构、实验室和检验机构等相关机构的认可工作。 在软件测试行业&#xff0c;CNAS认可具有重要意义。它标志着一个软件测试公…

站群服务器如何提高搜索引擎排名

站群服务器是一种专门为多个相关联的网站提供支持的服务器&#xff0c;旨在通过网站集合的形式提高搜索引擎排名和曝光度。那么站群服务器如何提高搜索引擎排名呢?Rak部落小编为您整理发布。 站群服务器提高搜索引擎排名的原理主要在于以下几个方面&#xff1a; - **提高网站…