【学习记录】PointLIO代码 update_iterated_dyn_share_modified 中函数指针的用法

本文主要是介绍【学习记录】PointLIO代码 update_iterated_dyn_share_modified 中函数指针的用法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近在看PointLio的代码,有一部分看了半天没看懂,学习整理如下。

1、PointLio在迭代卡尔曼部分的代码

esekfom.hpp中,有部分代码如下:

void init_dyn_share_modified(processModel f_in, processMatrix1 f_x_in, measurementModel_dyn_share_modified h_dyn_share_in)
{f = f_in;f_x = f_x_in;// f_w = f_w_in;h_dyn_share_modified_1 = h_dyn_share_in;maximum_iter = 1;x_.build_S2_state();x_.build_SO3_state();x_.build_vect_state();x_.build_SEN_state();
}

之后,在迭代更新时,部分代码如下:

bool update_iterated_dyn_share_modified() {dyn_share_modified<scalar_type> dyn_share;state x_propagated = x_;int dof_Measurement;double m_noise;for(int i=0; i<maximum_iter; i++){dyn_share.valid = true;// 调用了 h_dyn_share_modified_1 函数,这个函数的类型是 `measurementModel_dyn_share_modified`// measurementModel_dyn_share_modified 是 void (state &, dyn_share_modified<scalar_type> &) 这个类型函数的一个别名;// 具体的,这个函数的实现是: h_dyn_share_modified_1 指向的 `h_dyn_share_in`(在初始化init_dyn_share_modified(_2h)中)// 而 h_dyn_share_in 是 h_model_input,在 estimateor.cpp 中被定义了具体实现。h_dyn_share_modified_1(x_, dyn_share);if(! dyn_share.valid){return false;// continue;}// 省略后续代码...

比较费解的是,上述代码中,h_dyn_share_modified_1(x_, dyn_share);这一行到底是在做什么。
仔细去看定义,可以发现,具体定义如下:

typedef Eigen::Matrix<scalar_type, m, n> processMatrix1(state &, const input &);
typedef Eigen::Matrix<scalar_type, m, process_noise_dof> processMatrix2(state &, const input &);
typedef Eigen::Matrix<scalar_type, process_noise_dof, process_noise_dof> processnoisecovariance;
typedef void measurementModel_dyn_share_modified(state &, dyn_share_modified<scalar_type> &);

其实也看不懂,于是问了下GPT,知道了这是“函数指针”,但还是云里雾里;又翻开了《C++Primer》和查阅了一些网络资料。

2、函数指针

函数指针和指针函数,区别如下:
在这里插入图片描述
图片来自哔哩哔哩:指针函数与函数指针

概括来说,函数指针,是一个指针,指向一个函数。那么面临着一个问题,这个函数到底是什么类型?

2.1 函数的类型

之前一直没有关注过函数的“类型”,《C++ Primer》指出:
在这里插入图片描述
即函数应该是一个 bool (const string&, const string&) 类型,那么函数指针就是一个指向这种类型的函数的一个指针。函数名称可以是任何名字。

2.2 形参为指针的函数

由于函数指针,本质上是一个指针,那么这个指针,可以当作另一个函数的形参,如下:
在这里插入图片描述
此时,useBigger这个函数接收的第3个参数是一个指针,这个指针指向了一个函数而已。

但是这种写法比较复杂,因此引入typedef关键字:

2.3 typedef 关键字

这里需要提一下typedef关键字。typedef是类型定义的替换,有一个视频讲的很好:
https://www.bilibili.com/video/BV15h4y1M7p2

在这里插入图片描述
即“用变量名替换上面typedef后面的类型,然后去掉typedef,剩下的就是这一行的定义”。

所以,再结合上面形参是指针的函数,可以写作这样:
在这里插入图片描述
我们把Func换成上方绿色的typedef定义,就能看懂了。

3、回到代码

下面回到PointLio中的代码,我们捋一下思路:

init_dyn_share_modified函数:

在初始化时,将h_dyn_share_modified_1这个“函数指针”和传入的形参h_dyn_share_in共享,即指向的都是h_dyn_share_in这个形参所代表的函数。

init_dyn_share_modified这个函数的调用,是在laserMapping.cpp中:

kf_input.init_dyn_share_modified(get_f_input, df_dx_input, h_model_input);

即传入的参数是指向h_model_input的指针,而这个h_model_inputEstimator.cpp中定义,就是“IMU做输入量的观测模型”(这里不展开解释,IMU的“输入”/“输出”的区别)。

update_iterated_dyn_share_modified函数:

那么,再往下,初始化完成以后,h_dyn_share_modified_1就指向了h_model_input这个函数,
在接下来update_iterated_dyn_share_modified函数中,h_dyn_share_modified_1(x_, dyn_share)就是相当于调用了h_model_input这个函数。

可以再去看h_model_input这个函数,和h_dyn_share_modified_1函数,传入的形参形式是一致的。所以,我理解的应该是正确的吧。

至此,搞明白了这部分的调用。

小彩蛋

当在代码中自己添加注释,解释这部分函数指针时,我想说“写的真复杂”,结果github的copilot自动补全了我的注释,如下:
在这里插入图片描述看样子可能不止我一个人被这个函数指针的写法绕进去过吧。

这篇关于【学习记录】PointLIO代码 update_iterated_dyn_share_modified 中函数指针的用法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/788625

相关文章

全面解析Golang 中的 Gorilla CORS 中间件正确用法

《全面解析Golang中的GorillaCORS中间件正确用法》Golang中使用gorilla/mux路由器配合rs/cors中间件库可以优雅地解决这个问题,然而,很多人刚开始使用时会遇到配... 目录如何让 golang 中的 Gorilla CORS 中间件正确工作一、基础依赖二、错误用法(很多人一开

java中pdf模版填充表单踩坑实战记录(itextPdf、openPdf、pdfbox)

《java中pdf模版填充表单踩坑实战记录(itextPdf、openPdf、pdfbox)》:本文主要介绍java中pdf模版填充表单踩坑的相关资料,OpenPDF、iText、PDFBox是三... 目录准备Pdf模版方法1:itextpdf7填充表单(1)加入依赖(2)代码(3)遇到的问题方法2:pd

Java Stream流之GroupBy的用法及应用场景

《JavaStream流之GroupBy的用法及应用场景》本教程将详细介绍如何在Java中使用Stream流的groupby方法,包括基本用法和一些常见的实际应用场景,感兴趣的朋友一起看看吧... 目录Java Stream流之GroupBy的用法1. 前言2. 基础概念什么是 GroupBy?Stream

Java Spring的依赖注入理解及@Autowired用法示例详解

《JavaSpring的依赖注入理解及@Autowired用法示例详解》文章介绍了Spring依赖注入(DI)的概念、三种实现方式(构造器、Setter、字段注入),区分了@Autowired(注入... 目录一、什么是依赖注入(DI)?1. 定义2. 举个例子二、依赖注入的几种方式1. 构造器注入(Con

详解MySQL中JSON数据类型用法及与传统JSON字符串对比

《详解MySQL中JSON数据类型用法及与传统JSON字符串对比》MySQL从5.7版本开始引入了JSON数据类型,专门用于存储JSON格式的数据,本文将为大家简单介绍一下MySQL中JSON数据类型... 目录前言基本用法jsON数据类型 vs 传统JSON字符串1. 存储方式2. 查询方式对比3. 索引

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

Zabbix在MySQL性能监控方面的运用及最佳实践记录

《Zabbix在MySQL性能监控方面的运用及最佳实践记录》Zabbix通过自定义脚本和内置模板监控MySQL核心指标(连接、查询、资源、复制),支持自动发现多实例及告警通知,结合可视化仪表盘,可有效... 目录一、核心监控指标及配置1. 关键监控指标示例2. 配置方法二、自动发现与多实例管理1. 实践步骤

全面掌握 SQL 中的 DATEDIFF函数及用法最佳实践

《全面掌握SQL中的DATEDIFF函数及用法最佳实践》本文解析DATEDIFF在不同数据库中的差异,强调其边界计算原理,探讨应用场景及陷阱,推荐根据需求选择TIMESTAMPDIFF或inte... 目录1. 核心概念:DATEDIFF 究竟在计算什么?2. 主流数据库中的 DATEDIFF 实现2.1

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串