一文搞懂PCL中自定义点云类型的构建与函数使用

上周猛男快乐开发时遇到个bug,要用pcl的函数对自定义的点云进行处理。一起解决问题时遇到了很多问题,解决后整理出来分享给各位参考,以免踩一样的坑😊。文章中自定义的点我用PointT来表示,自定义点云一般指的是pcl::PointCloud<PointXYZITNormalVelocity> cloud

(修这个bug真的烧了很多脑子🧠)

文章目录

  • 1. 构造自定义PointT类型
  • 2. 用PCL的函数处理自定义点云
  • 3. 解决PCL和OpenCV同时使用导致的"no member named 'serialize' "问题
  • 4. 小结

1. 构造自定义PointT类型

为了涵盖日常开发中可能使用的点的类型,PCL中已经定义了很多种数据类型。因此在确定要构造自定义类型时,可以先去头文件pcl/impl/point_types.hpp里查看下是否有满足自己需求的点类型。

利用下面这个模板可以完成自定义PointT类型的定义,其中EIGEN_ALIGN16PCL_MAKE_ALIGNED_OPERATOR_NEW切记要加上。PCL中大量利用SSE指令集来加速,所以内存对齐是很必要的。

struct EIGEN_ALIGN16 _PointT // EIGEN_ALIGN16表示16字节对齐,[必须项]
{
  // 添加自定义点内部的字段信息  PCL_ADD_RGB;
  // ...
  
  // 必须项,确保内存正确对其,旧版本是EIGEN_MAKE_ALIGNED_OPERATOR_NEW 
  PCL_MAKE_ALIGNED_OPERATOR_NEW
};

struct EIGEN_ALIGN16 PointT: public _PointT
{
  // 参考下面的方式补充构造参数,当输入参数是constexpr,那么产生的对象的所有成员都是constexpr。
  // inline constexpr PointT (/*输入参数列表*/): /*初始化参数列表*/ {}
 
  // 重载运算符 <<,这样就可以通过std::cout << point 来查看点的信息。
  friend std::ostream& operator << (std::ostream& os, const PointT& p);
  
  PCL_MAKE_ALIGNED_OPERATOR_NEW
};

// 点云注册,这里的字段名影响PointCloud<PointT>等相关函数的拷贝、保存或加载功能。
POINT_CLOUD_REGISTER_POINT_STRUCT (_PointT,
    // 输入要注册的字段名,比如
    // (float, x, x)
    // (float, vx, vx)
)
POINT_CLOUD_REGISTER_POINT_WRAPPER(PointT, _PointT)

基于这个模板,我们来定义一个点PointXYZITNormalVelocity,其包含位置、法向、强度、速度、时间信息。

  • 第一步:定义基础类 _PointXYZITNormalVelocity。在添加字段时,参考如下规则:
    • 每组信息,满足16字节的优先构建。比如位置,法向,速度,这些数据组。其他信息比如强度,时间等放在后面定义。
    • 尽可能利用PCL提供的数据组构建方式。PCL中提供的数据组构建方式列举如下,一般16字节的数据组,都有个float [4]来填充,这个对应的字段名在定义的时候要记得防止冲突。切记:尽可能使用已有的,且不要尝试往字段的预留区域添加信息,比如位置信息PCL_ADD_POINT4D这个,pcl在做点云运算时,可能会将第四个字节也就是data[3]这里赋值为0或1来加速运算。
      • PCL_ADD_POINT4D [16字节]:定义字段float x,y,z。16字节对应的字段名为float data[4]
      • PCL_ADD_NORMAL4D [16字节]:定义字段float normal_x, normal_y, normal_z,或者可以通过字段float normal[3]来访问。16字节对应的字段名为float data_n[4]
      • PCL_ADD_RGB:定义字段uint8_t b, g, r, a,或者可以通过字段uint32_t rgba来访问。
      • PCL_ADD_INTENSITY:定义字段float intensity
      • PCL_ADD_INTENSITY_8U:定义字段uint8_t intensity
      • PCL_ADD_INTENSITY_32U:定义字段uint32_t intensity

基于上述信息,对应定义代码如下:

struct EIGEN_ALIGN16 _PointXYZITNormalVelocity
{
  PCL_ADD_POINT4D; // XYZ [16 bytes]
  PCL_ADD_NORMAL4D; // normal [16 bytes]
  union // Velocity [16 bytes]
  {
    float data_v[4];
    float velocity[3];
    struct
    {
      float v_x;
      float v_y;
      float v_z;
    };
  };
  PCL_ADD_INTENSITY;
  double time;
};
  • 第二步:构造最终点云类型PointXYZITNormalVelocity
    • 完善构造函数,补充几种可能用到的赋值情况。至少得定义个所有字段得赋值方式。
    • 补充运算符<<的重载。
struct EIGEN_ALIGN16 PointXYZITNormalVelocity: public _PointXYZITNormalVelocity
{
  // 几种可能用得到的构造函数,根据需求来定义即可
  inline constexpr PointXYZITNormalVelocity (const _PointXYZITNormalVelocity &p) :
      PointXYZITNormalVelocity {p.x, p.y, p.z, p.normal_x, p.normal_y, p.normal_z, 
      							p.v_x, p.v_y, p.v_z, p.intensity, p.time} {}
  inline constexpr PointXYZITNormalVelocity (float _x, float _y, float _z, 
  											 float _nx, float _ny, float _nz,
  											 float _vx, float _vy, float _vz, 
  											 float _intensity = 0.f, double _time = 0.0) :
      _PointXYZITNormalVelocity{{{_x, _y, _z, 1.0f}}, {{_nx, _ny, _nz, 0.0f}}, 
      							{{_vx, _vy, _vz}}, _intensity, _time} {}
  inline constexpr PointXYZITNormalVelocity (float _x, float _y, float _z):
      PointXYZITNormalVelocity (_x, _y, _z, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f) {}
  inline constexpr PointXYZITNormalVelocity (float _x, float _y, float _z,
  											 float _vx, float _vy, float _vz):
      PointXYZITNormalVelocity (_x, _y, _z, 0.0f, 0.0f, 0.0f, _vx, _vy, _vz) {}
  
  // 运算符重载
  friend std::ostream& operator << (std::ostream& os, const PointXYZITNormalVelocity& p)
  {
    os << "(" << "xyz: [" << p.x << "," << p.y << "," << p.z << "], ";
    os << "v: [" << p.v_x << "," << p.v_y << "," << p.v_z << "], ";
    os << "int: " << p.intensity << ", time: " << p.time << ")";
    return (os);
  }
  
  PCL_MAKE_ALIGNED_OPERATOR_NEW
};
  • 第三步:注册点云。这步是最简单得了,把所有需要用的字段按序注册即可
POINT_CLOUD_REGISTER_POINT_STRUCT (_PointXYZITNormalVelocity,
    (float, x, x)
    (float, y, y)
    (float, z, z)
    (float, normal_x, normal_x)
    (float, normal_y, normal_y)
    (float, normal_z, normal_z)
    (float, v_x, v_x)
    (float, v_y, v_y)
    (float, v_z, v_z)
    (float, intensity, intensity)
    (double, time, time)
)
POINT_CLOUD_REGISTER_POINT_WRAPPER(PointXYZITNormalVelocity, _PointXYZITNormalVelocity)

2. 用PCL的函数处理自定义点云

PCL很多函数是模板函数,但我们在使用PCL库时,动态库里面封装好的是预定义的点云类型的实现。任何用户代码都不需要编译模板化代码,从而加快了编译时间。

基于上述的定义方式,我们已经得到了一个点云pcl::PointCloud<PointXYZITNormalVelocity>::Ptr cloud。如果我们想直接调用函数
pcl::CropBox<PointXYZITNormalVelocity> crop时,会出现类似undefined reference to 'pcl::PCLBase<PointXYZITNormalVelocity>::setInputCloud(std::shared_ptr<pcl::PointCloud<PointXYZITNormalVelocity> const> const&)'的错误。

我分析了下代码细节,发现我们include的头文件里面的函数尽管是模板函数,但是函数实现部分并没有提供。函数细节对应的头文件放置在impl文件夹下。在include的头文件的最后,有个关键代码,即如果你没有定义PCL_NO_PRECOMPILE的话,编译代码时是无法获取函数的实现部分的,这也就会导致编译时出现undefined reference的问题。

在这里插入图片描述

知道了原因,解决起来就非常容易了,最好的办法是在项目的CMakeLists.txt里面添加add_definitions(-DPCL_NO_PRECOMPILE)。当然,如果项目不大的话也可以pcl头文件的前面添加#define PCL_NO_PRECOMPILE

除此之外,为了提高编译速度,我们还需要在定义点云的头文件的后面,跟上所使用函数的模板类显式实例化声明。(模板隐式实例化没啥问题,但会导致每个cpp编译时,文件较大,尽管link时候会删除,但会影响编译速度)

以我们要使用的pcl::CropBox为例,假如我们定义的点云放在头文件custom_types.h中,那么我们在该文件的末尾添加目标函数的实例化声明:template class pcl::CropBox<PointXYZITNormalVelocity>;。这样在实例化一次后,其他cpp引用这个无需重复实例化。

PS:各位对显示/隐式实例化感兴趣的可以参考另一个人的博客《C++11模板隐式实例化、显式实例化声明、定义》。

3. 解决PCL和OpenCV同时使用导致的"no member named ‘serialize’ "问题

再解决自定义点云的使用之后,编译项目时又出现了error: 'class std::unordered_map<unsigned int, std::vector<unsigned int>>' has no member name 'serialize'的错误。
在这里插入图片描述

经过大量的查阅资料,才搞懂问题原因,PCL和OpenCV都基于flann这个库,这个库
opencv是连接自己的flann库,而pcl是连接的系统的pcl库。
在这里插入图片描述
在这里插入图片描述
如果想正常编译,在调用pcl时必须得让USE_UNORDERED_MAP这个值为0。由于先调用的opencv,然后调用了pcl的库,进而触发了这个bug。

解决办法也很简单,在出问题的pcl头文件前,强行定义#define USE_UNORDERED_MAP 0即可(有的人是定义为1解决的,实际使用时候可以试试)。当然最优的办法是自己调整好头文件调用方式,把opencv和pcl的调用尽可能分开,然后在cpp里调用。但这样成本比较高,我们还是等待opencv优化下这个问题吧😆。
在这里插入图片描述
此外,我研究了下__GXX_EXPERIMENTAL_CXX0X__这个宏的来源。在Common-Predefined-Macros这里找到了对这个宏的说明。就是当编译时使用了c++11的特征时,即除了-std=C++98-std=gnu++98之外,这个宏就会被定义。这个宏已经过时了,在GCC 4.7.0中定义了__cplusplus宏,因此相关的代码最好用__cplusplus>=201103L这个方式来处理。
在这里插入图片描述

4. 小结

为了解决这个bug,特意去翻了PCL官网提供的教程《adding_custom_ptype》,讲得很不错,感兴趣可以看看。关于自定义点云中一些宏函数的理解,可以看看《PCL-common-定义点类型
》。GCC相关参数的理解还是得去看官方文档GCC文档。

你们猜猜我为啥分享这么多资料→_→,因为为了解决这个bug,我几乎把全网能翻的都翻了😭。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

SaaS智慧校园管理平台,智能电子班牌源码,智慧校园建设方案

电子班牌是什么&#xff1f; 电子班牌是一种智能交互终端&#xff0c;电子班牌可以解决“走班教学”考勤管理问题&#xff0c;将大数据、物联网和人工智能等新兴技术和教学管理工作融合&#xff0c;提升学校管理水平和管理效率。 电子班牌是在学校各教室门口安装高清可视化和具…

实验室管理系统 |基于springboot框架+ Mysql+JSP技术+Tomcat的实验室管理系统 设计与实现(可运行源码+数据库+设计文档)

推荐阅读100套最新项目 最新ssmjava项目文档视频演示可运行源码分享 最新jspjava项目文档视频演示可运行源码分享 最新Spring Boot项目文档视频演示可运行源码分享 目录 用户后台功能模块 用户后台管理 管理员功能登录前台功能效果图 系统功能设计 数据库E-R图设计 lunw…

论文阅读FCN-Transformer Feature Fusion for PolypSegmentation

本文提出了一种名为Fully Convolutional Branch-TransFormer (FCBFormer)的图像分割框架。该架构旨在结合Transformer和全卷积网络&#xff08;FCN&#xff09;的优势&#xff0c;以提高结肠镜图像中息肉的检测和分类准确性。 1&#xff0c;框架结构&#xff1a; 模型采用双分…

专题二 - 滑动窗口 - leetcode 904. 水果成篮 | 中等难度

leetcode 904. 水果成篮 leetcode 904. 水果成篮 | 中等难度1. 题目详情1. 原题链接2. 基础框架 2. 解题思路1. 题目分析2. 算法原理3. 时间复杂度 3. 代码实现4. 知识与收获 leetcode 904. 水果成篮 | 中等难度 1. 题目详情 你正在探访一家农场&#xff0c;农场从左到右种植…

【SQL Server】实验六 数据安全性

1 实验目的 掌握用户管理的基本方法&#xff0c;包括创建用户、删除用户和设置用户密码。掌握用户授权和回收权限的基本方法。掌握系统级权限和对象级权限的授权和回收方法掌握角色的使用方法 2 实验内容 2.1 掌握用户管理的基本使用方法 创建用户&#xff08;带密码&#…

RTP 控制协议 (RTCP) 反馈用于拥塞控制

摘要 有效的 RTP 拥塞控制算法&#xff0c;需要比标准 RTP 控制协议(RTCP)发送方报告(SR)和接收方报告(RR)数据包提供的关于数据包丢失、定时和显式拥塞通知 (ECN) 标记的更细粒度的反馈。 本文档描述了 RTCP 反馈消息&#xff0c;旨在使用 RTP 对交互式实时流量启用拥塞控制…

【每日力扣】235. 二叉搜索树的最近公共祖先与39. 组合总和问题描述

&#x1f525; 个人主页: 黑洞晓威 &#x1f600;你不必等到非常厉害&#xff0c;才敢开始&#xff0c;你需要开始&#xff0c;才会变的非常厉害。 235. 二叉搜索树的最近公共祖先 给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义…

简易版 RPC 框架实现 2.0 -netty实现

这一篇理解如果有难度&#xff0c;可能对netty不是很理解&#xff0c; 可以关注我netty专栏&#xff0c;还有另外一篇&#xff1a; 用 Netty 自己实现简单的RPC&#xff0c; 这一篇是学习netty的时候写的&#xff0c;更倾向于分析netty相关的知识&#xff0c; 今天我是学习dubb…

Delphi7应用教程学习1.3【练习题目】:文本及悬停文字的显示

这个例子主要用到了btn的Hint 属性&#xff0c;Hint是提示的意思。 还有Delphi7还是很好用的&#xff0c;改变了的属性是粗体&#xff0c;默认没有改变的属性为细体。

插入排序:一种简单而有效的排序算法

插入排序&#xff1a;一种简单而有效的排序算法 一、什么是插入排序&#xff1f;二、插入排序的步骤三、插入排序的C语言实现四、插入排序的性能分析五、插入排序的优化六、总结 在我们日常生活和工作中&#xff0c;排序是一种非常常见的操作。比如&#xff0c;我们可能需要对一…

B端界面又丑又乱,也不会总结规范,来,我给5个规范模板,照着学

发5个别人总结的规范&#xff0c;一定会对你的B端系统改进&#xff0c;有帮助的。

TSINGSEE青犀AI烟火识别等算法打造电瓶车消防安全解决方案

一、背景分析 根据国家消防救援局的统计&#xff0c;2023年全国共接报电动自行车火灾2.1万起&#xff0c;相比2022年上升17.4%&#xff0c;电动自行车火灾安全隐患问题不容忽视。 电瓶车火情主要问题和原因&#xff1a; 电瓶车/电池质量良莠不齐用户安全意识薄弱&#xff0c…

虚拟机NAT模式配置

注意这里IP要和网关在同一网段&#xff0c;且虚拟机默认网关末尾为.2&#xff08;如果默认网关配置为.1会与宿主机冲突&#xff0c;导致无法ping通外网&#xff09; 点击NAT模式下的NAT设置即可查看默认网关 这里的网关可以理解为主机与虚拟机交互的入口

蓝桥杯算法训练VIP-数组查找及替换

题目 1634: 蓝桥杯算法训练VIP-数组查找及替换 时间限制: 3s 内存限制: 192MB 提交: 1629 解决: 890 题目描述 给定某整数数组和某一整数b。要求删除数组中可以被b整除的所有元素&#xff0c;同时将该数组各元素按从小到大排序。如果数组元素数值在A到Z的ASCII之间&#xff0…

MySQL:SQL优化

1. 插入优化 使用insert语句单条单条数据插入效率偏低&#xff0c;建议使用insert批量插入数据&#xff0c;批量控制在500-1000条数据较为合适&#xff0c;当面对数以百万的数据时&#xff0c;可以使用load指令&#xff0c;提升插入数据效率 相关指令 #客户端连接服务端加上参…

Java后端面试经验分享,~纯分享

本文将从面试、工作、学习三个方面分享最近面试的一些心得以及以后发展的一些规划&#xff0c;仅供参考&#xff0c;哈哈&#xff0c;毕竟本人也很菜&#xff0c;因为菜才要多学习。一会儿也会分享两本Java面试题库&#xff08;题库是b站大学找的&#xff0c;一会儿我也会分享出…

C# Onnx C2PNet 图像去雾 室内场景

目录 介绍 效果 模型信息 项目 代码 下载 C# Onnx C2PNet 图像去雾 室内场景 介绍 github地址&#xff1a;GitHub - YuZheng9/C2PNet: [CVPR 2023] Curricular Contrastive Regularization for Physics-aware Single Image Dehazing [CVPR 2023] Curricular Contrasti…

DataGrip 面试题及答案整理,最新面试题

DataGrip的数据库兼容性和多数据库支持如何实现&#xff1f; DataGrip实现数据库兼容性和多数据库支持的方式包括&#xff1a; 1、广泛的数据库支持&#xff1a; DataGrip支持多种数据库&#xff0c;包括但不限于MySQL, PostgreSQL, SQL Server, Oracle, SQLite, 和MongoDB&a…

C++:类之六脉神剑——默认成员函数

个人主页&#xff1a;日刷百题 系列专栏&#xff1a;〖C/C小游戏〗〖Linux〗〖数据结构〗 〖C语言〗 &#x1f30e;欢迎各位→点赞&#x1f44d;收藏⭐️留言&#x1f4dd; ​ ​ 一、默认成员函数 如果一个类中什么成员都没有&#xff0c;简称为 空类 。 空类中真的什么都…

管理类联考–复试–政治--二十大--记忆宫殿

文章目录 整体记忆宫殿门床头柜床书桌阳台 口诀记忆法 整体 记忆宫殿 要有逻辑的放到房间了 何为逻辑&#xff0c;如下大佬总结的便是&#xff0c;或者可自行总结&#xff0c;有前后顺序&#xff0c;做事逻辑即可 第一步&#xff1a;将逻辑的点放到房间里的点&#xff0c;…