当前位置: 首页 > news >正文

点云匹配方法 NDT(正态分布变换)

1. 正态分布变换 (NDT)

在点云匹配中,ICP基于距离直接最优化变换矩阵的参数,由于是欠定方程且旋转矩阵的约束,使得结果很难优化,为此在新的维度优化变换矩阵的参数,被很好的提出:

先将参考点云(目标点云)转换为多维变量的正态分布,匹配的点云如果采用某组变换参数后,使得新的点云和目标点云的正态分布参数匹配很好,那么变换点在参考系中的概率密度将会很大。因此,可以考虑用优化的方法求出使得概率密度之和最大的变换参数,此时两幅激光点云数据将匹配的最好。

2. 多元正态分布

我们知道,如果随机变量X满足正态分布,则其概率密度函数为:

对于多元正态分布而言,其概率密度函数可以表示为:

2.1. NDT算法流程

将参考点云网格化,计算每个网格的概率密度函数:

网格的概率密度函数则为:

2.2. 变换参数和最大似然

我们需要优化的参数就是对当前点云的坐标变换(旋转,平移等),转换函数表示使用姿态变换来变换,结合之前的一组状态密度函数,那么最好的变换参数应该是最大化似然函数的姿态变换:

那么最大化似然也就相当于最小化负对数似然

就到了我们最熟悉的最优化的部分了。

现在的任务就是使用优化算法来调整变换参数来最小化这个负对数似然。NDT算法使用牛顿法进行参数优化。

3.算法流程

输入:两帧点云数据,

输出:位姿变换参数 R, t

a. 对计算正太分布变换 (NDT);
b. 初始化需要估计的参数 R, t,设置为0或者采用里程计数据;
c. 根据参数 R, t 将中的所有样本点投影到对应的坐标系下;
d. 确定每一个映射点对应的正态分布即映射点属于中的哪一个cell;
e. 通过评估每个映射点的分布并对结果求和来确定参数的得分 score;
f. 通过尝试优化分数来计算新的参数估计。这是通过执行牛顿算法(非线性最小二乘中一种方法)的一个步骤来实现的;
g. 转到c,直到满足收敛条件。

4. 代码

#include <iostream> #include <thread> #include <pcl/io/pcd_io.h> #include <pcl/point_types.h> #include <pcl/registration/ndt.h> #include <pcl/filters/approximate_voxel_grid.h> #include <pcl/visualization/pcl_visualizer.h> using namespace std::chrono_literals; int main () { // Loading first scan of room. pcl::PointCloud<pcl::PointXYZ>::Ptr target_cloud (new pcl::PointCloud<pcl::PointXYZ>); if (pcl::io::loadPCDFile<pcl::PointXYZ> ("room_scan1.pcd", *target_cloud) == -1) { PCL_ERROR ("Couldn't read file room_scan1.pcd \n"); return (-1); } std::cout << "Loaded " << target_cloud->size () << " data points from room_scan1.pcd" << std::endl; // Loading second scan of room from new perspective. pcl::PointCloud<pcl::PointXYZ>::Ptr input_cloud (new pcl::PointCloud<pcl::PointXYZ>); if (pcl::io::loadPCDFile<pcl::PointXYZ> ("room_scan2.pcd", *input_cloud) == -1) { PCL_ERROR ("Couldn't read file room_scan2.pcd \n"); return (-1); } std::cout << "Loaded " << input_cloud->size () << " data points from room_scan2.pcd" << std::endl; // Filtering input scan to roughly 10% of original size to increase speed of registration. pcl::PointCloud<pcl::PointXYZ>::Ptr filtered_cloud (new pcl::PointCloud<pcl::PointXYZ>); pcl::ApproximateVoxelGrid<pcl::PointXYZ> approximate_voxel_filter; approximate_voxel_filter.setLeafSize (0.2, 0.2, 0.2); approximate_voxel_filter.setInputCloud (input_cloud); approximate_voxel_filter.filter (*filtered_cloud); std::cout << "Filtered cloud contains " << filtered_cloud->size () << " data points from room_scan2.pcd" << std::endl; // Initializing Normal Distributions Transform (NDT). pcl::NormalDistributionsTransform<pcl::PointXYZ, pcl::PointXYZ> ndt; // Setting scale dependent NDT parameters // Setting minimum transformation difference for termination condition. ndt.setTransformationEpsilon (0.01); // Setting maximum step size for More-Thuente line search. ndt.setStepSize (0.1); //Setting Resolution of NDT grid structure (VoxelGridCovariance). ndt.setResolution (1.0); // Setting max number of registration iterations. ndt.setMaximumIterations (35); // Setting point cloud to be aligned. ndt.setInputSource (filtered_cloud); // Setting point cloud to be aligned to. ndt.setInputTarget (target_cloud); // Set initial alignment estimate found using robot odometry. Eigen::AngleAxisf init_rotation (0.6931, Eigen::Vector3f::UnitZ ()); Eigen::Translation3f init_translation (1.79387, 0.720047, 0); Eigen::Matrix4f init_guess = (init_translation * init_rotation).matrix (); // Calculating required rigid transform to align the input cloud to the target cloud. pcl::PointCloud<pcl::PointXYZ>::Ptr output_cloud (new pcl::PointCloud<pcl::PointXYZ>); ndt.align (*output_cloud, init_guess); std::cout << "Normal Distributions Transform has converged:" << ndt.hasConverged () << " score: " << ndt.getFitnessScore () << std::endl; // Transforming unfiltered, input cloud using found transform. pcl::transformPointCloud (*input_cloud, *output_cloud, ndt.getFinalTransformation ()); // Saving transformed input cloud. pcl::io::savePCDFileASCII ("room_scan2_transformed.pcd", *output_cloud); // Initializing point cloud visualizer pcl::visualization::PCLVisualizer::Ptr viewer_final (new pcl::visualization::PCLVisualizer ("3D Viewer")); viewer_final->setBackgroundColor (0, 0, 0); // Coloring and visualizing target cloud (red). pcl::visualization::PointCloudColorHandlerCustom<pcl::PointXYZ> target_color (target_cloud, 255, 0, 0); viewer_final->addPointCloud<pcl::PointXYZ> (target_cloud, target_color, "target cloud"); viewer_final->setPointCloudRenderingProperties (pcl::visualization::PCL_VISUALIZER_POINT_SIZE, 1, "target cloud"); // Coloring and visualizing transformed input cloud (green). pcl::visualization::PointCloudColorHandlerCustom<pcl::PointXYZ> output_color (output_cloud, 0, 255, 0); viewer_final->addPointCloud<pcl::PointXYZ> (output_cloud, output_color, "output cloud"); viewer_final->setPointCloudRenderingProperties (pcl::visualization::PCL_VISUALIZER_POINT_SIZE, 1, "output cloud"); // Starting visualizer viewer_final->addCoordinateSystem (1.0, "global"); viewer_final->initCameraParameters (); // Wait until visualizer window is closed. while (!viewer_final->wasStopped ()) { viewer_final->spinOnce (100); std::this_thread::sleep_for(100ms); } return (0); }

参考文献

自动驾驶(十)---------正态分布变换(NDT)_一实相印的博客-CSDN博客

NDT点云配准_有了个相册的博客-CSDN博客_ndt点云配准

3d激光雷达开发(ndt匹配)_嵌入式-老费的博客-CSDN博客_ndt匹配

http://www.jsqmd.com/news/813281/

相关文章:

  • 珠三角保安公司甄选指南!惠州东莞深圳广州佛山保安公司对比,惠州工厂保安公司优选认准广东国卫保安 - 栗子测评
  • yargs单元测试终极指南:使用mocha测试CLI命令的完整实践
  • 杭州锦纪财务咨询有限公司2026一站式财税优选:杭州工商注册/记账报税/代办营业执照/出口退税代理推荐杭州锦纪财务 - 栗子测评
  • 基于AutoHotkey v2的Cursor AI编程效率工具:CapsLock快捷键方案详解
  • 从绿度到热度:拆解RSEI遥感生态指数的四个核心指标(GEE/Landsat 8版)
  • 【独家首发】DeepSeek-V2模型GPU利用率可视化方案:仅需3个自定义Metrics,告别盲调参数
  • PX4-Autopilot扩展卡尔曼滤波状态估计系统深度解析与实战调优
  • 深入解析Baichuan-7B:从GPT架构到LoRA微调的实践指南
  • 2026年卫生级拉丝白钢板/316L白钢板/321白钢板/沈阳拉丝白钢板推荐厂家精选 - 品牌宣传支持者
  • 2026年测试工程师常用性能测试平台:高效办公与场景适配指南
  • 【坐标转换实战】从公式到代码:极坐标与笛卡尔坐标互通的编程实现与象限陷阱
  • 联发科与威睿电通合作:深度解析全球模式SoC如何实现CDMA与LTE融合
  • 三步轻松上手:BilldDesk Pro开源远程桌面控制工具完整指南
  • 2026年4月正规的稀有金属回收企业推荐,物资回收/废旧物资回收/废旧金属回收/工程材料回收,稀有金属回收服务厂家选哪家 - 品牌推荐师
  • 中国词元与世界AI元语:模力方舟和口袋龙虾的协同进化
  • FPGA实战:在ZCU102上构建MIG控制器与DDR4通信的完整流程
  • 深入浅出:用Grad-CAM解锁Swin Transformer的视觉注意力
  • educoder--网络实验--Wireshark实战:NAT地址转换全流程解析
  • 基于Tailwind CSS的Skeleton UI组件库:现代Web开发的高效解决方案
  • 提升boardgame.io游戏加载速度的终极指南:客户端缓存策略全解析
  • 【软考网络工程师综合分析题整理(2026.5.13)】
  • 量子支持向量机原理与硬件优化实践
  • 从专利大国到专利强国:企业全球专利布局策略与实战指南
  • 锌铝合金产品定制厂家推荐:2026锌合金铝合金零配件压铸+金属件电镀喷涂加工厂权威推荐 - 栗子测评
  • ARM错误恢复中断机制与ERRERICR2寄存器详解
  • Sutton《苦涩的教训》早已预言:一切**人工精巧设计的专用智能系统**,终将被算力与数据驱动的通用范式无情取代
  • 在Windows上构建GTK应用:从环境搭建到首个跨平台GUI
  • STM32F407实战:从SWD/JTAG电路设计到ST-LINK避坑指南
  • Dyon 4D向量与矩阵:游戏开发与图形编程的终极利器
  • 2026年工业级拉丝白钢板/310s白钢板/耐高温白钢板批量采购厂家推荐 - 行业平台推荐