VTK——使用ICP算法进行模型配准

ICP算法

迭代最近点(Iterative Closest Point,ICP)算法是一种用于两个三维形状之间几何对齐(也叫做配准)的计算方法。通常,这两个形状至少有一个是点云数据。ICP算法用于最小化源点云与目标点云之间点到点的距离,从而寻找两者之间的最佳匹配变换(通常是刚体变换,但也可能是仿射或其他形式的变换)。

算法流程

初始化:设定初始猜测的变换参数(通常为单位矩阵)。
关联点:对于源点云中的每一个点,找到目标点云中的最近点。
最小化误差:找到一个变换,该变换能最小化源点到对应目标点之间的距离。
更新变换:将找到的变换应用于源点云。
收敛检查:检查误差或变换是否在一定阈值内收敛。如果是,则算法停止。否则,返回步骤2。

数学公式

设源点云 P 和目标点云Q 由 n 个点组成,其中pi​∈P 和 qi​∈Q。我们想要找到一个变换 T,使得总体误差 E 最小。
在这里插入图片描述
对于刚体变换,T 可以由旋转矩阵 R 和平移矢量 t 组成:
在这里插入图片描述
这个算法的优点是简单、直观、实现相对容易,缺点是容易收敛到局部最优解和计算复杂性较高。经常应用到机器人导航、对象识别、三维建模等场景。

代码流程

  1. 计算模型的质心:使用 VTK 的 vtkCenterOfMass 类计算 STL 模型的质心。这个质心用于后续找出模型中距质心最远的几个点。

  2. 找出四个距离质心最远的点:迭代遍历模型所有的点,并使用 VTK 的 vtkMath::Distance2BetweenPoints 方法计算每个点与质心的距离。找出四个距离质心最远的点,并保存它们的索引。

  3. 创建一个新的 vtkPolyData 对象,包含这四个点:使用 vtkPoints 和 vtkCellArray 创建一个新的 vtkPolyData 对象,这个对象只包括距离质心最远的四个点。

  4. 创建另一个 vtkPolyData 对象,包含在 3D 模型中选定的点:这个对象是算法的目标点集,也是使用 vtkPoints 和 vtkCellArray 创建的。

  5. 应用 ICP 算法:使用 VTK 的 vtkIterativeClosestPointTransform 类进行 ICP 对齐。设置源点(STL模型中的四个点)和目标点(3D模型中选定的点),并启动算法。

  6. 应用变换到 STL 模型:通过 vtkTransformPolyDataFilter 将 ICP 算法得出的变换应用到原始的 STL 模型上。

  7. 更新渲染:最后,更新模型的渲染,以反映应用了 ICP 变换后的新状态。

代码的核心是使用 ICP 算法进行两组点之间的最优对齐。它首先选取 STL 模型中四个特定的点(距离质心最远的点),然后用这些点与 3D 模型中预先选定的点进行对齐。通过这种方式,算法能够找到一个最佳的刚体变换,将 STL 模型与 3D 模型对齐。在这里面,所做的只是一个简单的配准示例。因为两个模型特征点的选取和ICP算法的应用都是比较复杂的,需要不断的尝试和优化,才能得到一个比较好的效果。

void performModelAlignment() {// Calculate Model Centerdouble modelCenter[3];auto centerCalculator = vtkSmartPointer<vtkCenterOfMass>::New();centerCalculator->SetInputData(reader->GetOutput());centerCalculator->SetUseScalarsAsWeights(false);centerCalculator->Update();centerCalculator->GetCenter(modelCenter);// Find Distant Points from Centerauto modelPoints = reader->GetOutput()->GetPoints();std::vector<vtkIdType> distantPoints = findDistantPoints(modelPoints, modelCenter, 4);// Construct PolyData for Distant Points in STLauto stlSelectedPointsData = createPolyDataFromPoints(modelPoints, distantPoints);// Construct PolyData for Selected Points in 3D Modelauto modelSelectedPointsData = createPolyDataFromPoints(featurePoints);// Perform ICPauto icpTransform = vtkSmartPointer<vtkIterativeClosestPointTransform>::New();icpTransform->SetSource(stlSelectedPointsData);icpTransform->SetTarget(modelSelectedPointsData);icpTransform->GetLandmarkTransform()->SetModeToRigidBody();icpTransform->Modified();icpTransform->Update();// Apply TransformationapplyTransformationToModel(icpTransform, reader->GetOutput());}std::vector<vtkIdType> findDistantPoints(vtkSmartPointer<vtkPoints> points, double center[3], int numPoints) {std::vector<vtkIdType> distantPoints;// ... (Same logic to find distant points)return distantPoints;
}vtkSmartPointer<vtkPolyData> createPolyDataFromPoints(vtkSmartPointer<vtkPoints> points, std::vector<vtkIdType> &selectedIds) {auto polyData = vtkSmartPointer<vtkPolyData>::New();auto selectedPoints = vtkSmartPointer<vtkPoints>::New();auto vertices = vtkSmartPointer<vtkCellArray>::New();// ... (Same logic to create PolyData)return polyData;
}void applyTransformationToModel(vtkSmartPointer<vtkIterativeClosestPointTransform> icpTransform, vtkSmartPointer<vtkPolyData> originalData) {auto transformFilter = vtkSmartPointer<vtkTransformPolyDataFilter>::New();transformFilter->SetInputData(originalData);transformFilter->SetTransform(icpTransform);transformFilter->Update();vtkPolyDataMapper::SafeDownCast(m_modelActor->GetMapper())->SetInputData(transformFilter->GetOutput());
}

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

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

相关文章

java企业工程管理系统源码之提高工程项目管理软件的效率

高效的工程项目管理软件不仅能够提高效率还应可以帮你节省成本提升利润 在工程行业中&#xff0c;管理不畅以及不良的项目执行&#xff0c;往往会导致项目延期、成本上升、回款拖后&#xff0c;最终导致项目整体盈利下降。企企管理云业财一体化的项目管理系统&#xff0c;确保…

Golang设计模式

Golang设计模式 Golang设计模式简介Golang工厂设计模式Golang单例设计模式Golang抽象工厂设计模式Golang建造者模式 (Builder Pattern)Golang 原型模式(Prototype Pattern)Golang适配器模式Golang 桥接模式&#xff08;Bridge Pattern&#xff09;Golang装饰器模式(Decorator …

Qt5界面Qt Designer上添加资源图片后,ModuleNotFoundError: No module named ‘rcc_rc‘ 的终极解决方案

在网上找了很久都没弄明白&#xff0c;最后还是自己思考解决了。 起因&#xff1a; 用 Qt Designer 添加资源文件作为背景图&#xff0c;编译 \resource\static\qrc> pyuic5 -o .\xx.py .\xx.ui发现在 xx.py 文件末尾中多了一个语句&#xff1a; import rcc_rc然后运行就…

vue项目——表情选择器

组件库地址&#xff1a;https://www.npmjs.com/package/emoji-mart-vue 1、下载 npm install --save emoji-mart-vue 2、引入 import { Picker } from emoji-mart-vueexport default {components: {Picker} }3、使用 <picker set"emojione" /> <picker …

ChatGPT 总结前端HTML, JS, Echarts都包含哪些内容

AIGC ChatGPT ,BI商业智能, 可视化Tableau, PowerBI, FineReport, 数据库Mysql Oracle, Office, Python ,ETL Excel 2021 实操,函数,图表,大屏可视化 案例实战 http://t.csdn.cn/zBytu

M1 Pro 新芯片安装python2 方案汇总

前言&#xff1a;磨刀不误砍柴工&#xff0c;环境装好&#xff0c;才能打工。M1 Pro 新芯片安装python2 文章目录 方案一 docker 容器构造环境&#xff08;如果涉及本地两个仓库需要关联则不适用&#xff09;方案二 使用 pyenv &#x1f680; 作者简介&#xff1a;作为某云服务…

thinkphp6 入门(3)--获取GET、POST请求的参数值

一、Request对象 thinkphp提供了Request对象&#xff0c;其可以 支持对全局输入变量的检测、获取和安全过滤 支持获取包括$_GET、$_POST、$_REQUEST、$_SERVER、$_SESSION、$_COOKIE、$_ENV等系统变量&#xff0c;以及文件上传信息 具体参考&#xff1a;https://www.kanclou…

Fedora Linux Flatpak 八月推荐应用

导读本文介绍了 Flathub 中可用的项目以及安装说明。 Flathub 是获取和分发适用于所有 Linux 的应用的地方。它由 Flatpak 提供支持&#xff0c;允许 Flathub 应用在几乎任何 Linux 发行版上运行。 请阅读 “Flatpak 入门”。要启用 Flathub 作为你的 Flatpak 提供商&#xff…

什么是jvm

一、初识JVM&#xff08;虚拟机&#xff09; JVM是Java Virtual Machine&#xff08;Java虚拟机&#xff09;的缩写&#xff0c;JVM是一种用于计算设备的规范&#xff0c;它是一个虚构出来的计算机&#xff0c;是通过在实际的计算机上仿真模拟各种计算机功能来实现的。 引入Jav…

6种超简单方法告诉你卧室墙面该怎么去装饰

创意 1&#xff1a;夏日宁静 潮汐的变化并不意味着你的夏日精神也要随之改变。用海洋壁画装饰海滩吧&#xff0c;让咸咸的空气、沙滩上的脚趾和木板道上的梦想随风飘散。 创意 2&#xff1a;漂浮艺术 如果你是那种喜欢少走弯路的人&#xff0c;可以考虑完全避开 “墙面艺术&qu…

pytorch中nn.Conv1d功能介绍

在使用Conv1d函数时&#xff0c;pytorch默认你的数据是一维的&#xff0c;比如一句话“深度学习”可以用一个一维数组 [深, 度, 学, 习] 表示&#xff0c;这个数据就是一维的。图片是二维数据&#xff0c;它有长宽两个维度。 因此在使用 Conv1d 函数时&#xff0c;输入是一个三…

14:00面试,14:08就出来了,问的问题有点变态

从小厂出来&#xff0c;没想到在另一家公司又寄了。 到这家公司开始上班&#xff0c;加班是每天必不可少的&#xff0c;看在钱给的比较多的份上&#xff0c;就不太计较了。没想到8月一纸通知&#xff0c;所有人不准加班&#xff0c;加班费不仅没有了&#xff0c;薪资还要降40%,…

小白学Go 基础02-了解Go语言的诞生与演进

Go语言诞生于何时&#xff1f;它的最初设计者是谁&#xff1f;它为什么被命名为Go&#xff1f;它的设计目标是什么&#xff1f;它如今发展得怎么样&#xff1f;带着这些问题&#xff0c;我们一起穿越时空&#xff0c;回到2007年9月Go语言诞生的那一历史时刻吧。 Go语言的诞生 …

数据结构与算法基础-学习-30-插入排序之直接插入排序、二分插入排序、希尔排序

一、排序概念 将一组杂乱无章的数据按一定规律顺次排列起来。 将无序序列排成一个有序序列&#xff08;由小到大或由大到小&#xff09;的运算。 二、排序方法分类 1、按数据存储介质 名称描述内部排序数据量不大、数据在内存&#xff0c;无需内外交换存交换存储。外部排序…

【CLIP详读】

个人网站&#xff1a;https://tianfeng.space 一、前言 OpenAI的CLIP项目自从推出以来&#xff0c;CLIP引起了广泛的关注。它的方法看似简单&#xff0c;但效果非常出色&#xff0c;许多结果令人惊叹。例如&#xff0c;预训练模型可以在任何视觉分类数据集上实现出色的效果&a…

Elasticsearch 集成--Flink 框架集成

一、Flink 框架介绍 Apache Spark 是一种基于内存的快速、通用、可扩展的大数据分析计算引擎。 Apache Spark 掀开了内存计算的先河&#xff0c;以内存作为赌注&#xff0c;赢得了内存计算的飞速发展。 但是在其火热的同时&#xff0c;开发人员发现&#xff0c;在 Spark …

【Linux】DNS系统,ICMP协议,NAPT技术

遏制自己内心的知识优越感&#xff0c;才能让你发自内心的去尊重他人&#xff0c;避免狂妄自大&#xff0c;才能让你不断的丰富自己的内心。 文章目录 一、DNS系统1.DNS服务器返回域名对应的ip2.使用dig工具分析DNS过程3.浏览器中输入url后发生的事情&#xff1f; 二、ICMP协议…

重磅OpenAI发布ChatGPT企业版本

8月29日凌晨&#xff0c;Open AI官网发布ChatGPT企业版本&#xff01; 企业版简介&#xff1a; ChatGPT企业版提供企业级安全和隐私、无限的高速 GPT-4 访问、用于处理更长输入的更长上下文窗口、高级数据分析功能、自定义选项等等。人工智能可以协助和提升我们工作生活的各个…

postgresql-条件表达式

postgresql-条件表达式 简单Case表达式搜索Case表达式缩写函数总结 简单Case表达式 select e.first_name , e.last_name , e.department_id , case e.department_id when 90 then 管理when 60 then 开发else 其他end as "部门" from cps.public.employees e ;-- 统…

第一课:使用C++实现图片去水印

1.功能概述 实现图片去水印的方法有很多,下面提供一种基于OpenCV库的C++实现方法。主要思路是利用图像中不同水印区域之间的差异,进行区域提取、重构和合成,从而实现去除水印的效果。 2.具体实现 2.1.导入OpenCV库和头文件 #include <iostream> #include <o…