简单形状点云轮廓点排序(旋转角)

1、背景介绍

      很多边缘提取算法提取的边缘点为无序点云,如下图所示,无序点云不利于后续各种应用,比如根据边缘计算点云面积、点云轮廓线规则化等。若对点云进行排序,则可以进行上述引用。但实际上,点云形状错综复杂,对点云进行排序难度会比较大。此次博客介绍一种针对简单形状点云的边缘点排序原理,对于复杂形状点云,仅供参考。

图 1 边缘点无序与有序示例

2、原理介绍

      对于形状简单的点云,可以使用与某一方向的夹角,按照夹角大小进行排序,实现点的排序,如下图所示。

图 2 轮廓点与起始方向夹角示意图

对点进行排序的步骤如下:

(1)中心计算:统计所有点的平均坐标作为中心

(2)计算旋转角:以y轴为起始方向,计算每个轮廓点与y轴的夹角,得到夹角集合

(3)角度排序:对角度从小到大进行排序,按照排序后的角度对轮廓点进行排序,即实现角度排序

****在对角度排序时,使用for循环找到角度停止即可,可以提高排序效率。

for (int i = 0; i < sortedangles.size(); i++)//依据角度寻找点{for (int j = 0; j < allangles.size(); j++){if (sortedangles[i] == allangles[j]){result.push_back(unorderpt[j]);break;//终止}}}

3、测试

       源代码下载链接:简单形状点云轮廓点排序(旋转角)-CSDN博客   

 本程序在PCL配置好的环境下进行运行,只需要将配置好的环境下,将利用旋转角对点排序.cpp添加到源文件目录下运行即可。

3.1 可视化未排序前结果

测试主函数代码如下:

//未排序前的点,进行连接
void main()
{//(1)读入边缘点vector<pcl::PointXYZ> boundpts;string filepath = "D:\\testdata\\testdata_01_boud.txt";std::ifstream pointsFile(filepath, std::ios::in);pcl::PointXYZ point;char line[128];double x, y, z;int r, g, b;while (pointsFile.getline(line, sizeof(line))){std::stringstream word(line);word >> x;word >> y;word >> z;word >> r;word >> g;word >> b;point.x = x;point.y = y;point.z = z;if (r == 255 && g == 0 && b == 0)//边缘点(为红色点){pcl::PointXYZ onept;onept.x = x;onept.y = y;onept.z = z;boundpts.push_back(onept);}}pcl::visualization::PCLVisualizer viewer("point connect");viewer.setBackgroundColor(0, 0, 0);//(2)可视化pcl::PointCloud<pcl::PointXYZ>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZ>);cloud->width = boundpts.size();cloud->height = 1;cloud->is_dense = false;cloud->resize(cloud->width*cloud->height);for (int i = 0; i < cloud->width; i++){cloud->points[i].x = boundpts[i].x;cloud->points[i].y = boundpts[i].y;cloud->points[i].z = boundpts[i].z;}//(3)增加点pcl::visualization::PointCloudColorHandlerCustom<pcl::PointXYZ> single_color(cloud, 255, 0, 0);viewer.addPointCloud<pcl::PointXYZ>(cloud, single_color, "sample cloud");viewer.setPointCloudRenderingProperties(pcl::visualization::PCL_VISUALIZER_POINT_SIZE, 3, "sample cloud");//(4)增加多边形pcl::PlanarPolygon<pcl::PointXYZ> polygon;pcl::PointCloud<pcl::PointXYZ> contour;contour.width = boundpts.size();contour.height = 1;contour.is_dense = false;contour.resize(contour.height*contour.width);for (int i = 0; i < boundpts.size(); i++){contour.points[i] = boundpts[i];}polygon.setContour(contour);viewer.addPolygon(polygon, 0, 255, 0, "ploygon", 0);while (!viewer.wasStopped()){viewer.spinOnce(1);}
}

无序点连接可视化结果如下,可以发现点之间无序,其顺序与原始点云文件中点的顺序有关,影响厚后续处理。

   

     

3.2 可视化排序后点

测试主函数代码如下:


//测试排序后点,将其连接起来
void main()
{//(1)读入边缘点vector<pcl::PointXYZ> boundpts;string filepath = "D:\\testdata\\testdata_01_boud.txt";std::ifstream pointsFile(filepath, std::ios::in);pcl::PointXYZ point;char line[128];double x, y, z;int r, g, b;while (pointsFile.getline(line, sizeof(line))){std::stringstream word(line);word >> x;word >> y;word >> z;word >> r;word >> g;word >> b;point.x = x;point.y = y;point.z = z;if (r == 255 && g == 0 && b == 0)//边缘点(为红色点){pcl::PointXYZ onept;onept.x = x;onept.y = y;onept.z = z;boundpts.push_back(onept);}}//(2)对边缘点排序vector<pcl::PointXYZ> orderpts = Orderpts(boundpts);//(3)对点进行连接,测试点是否排序了pcl::visualization::PCLVisualizer viewer("point connect");viewer.setBackgroundColor(0, 0, 0);//(3)可视化pcl::PointCloud<pcl::PointXYZ>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZ>);cloud->width = orderpts.size();cloud->height = 1;cloud->is_dense = false;cloud->resize(cloud->width*cloud->height);for (int i = 0; i < cloud->width; i++){cloud->points[i].x = orderpts[i].x;cloud->points[i].y = orderpts[i].y;cloud->points[i].z = orderpts[i].z;}//(4)增加点pcl::visualization::PointCloudColorHandlerCustom<pcl::PointXYZ> single_color(cloud, 255, 0, 0);viewer.addPointCloud<pcl::PointXYZ>(cloud, single_color, "sample cloud");viewer.setPointCloudRenderingProperties(pcl::visualization::PCL_VISUALIZER_POINT_SIZE, 3, "sample cloud");//(5)增加多边形pcl::PlanarPolygon<pcl::PointXYZ> polygon;pcl::PointCloud<pcl::PointXYZ> contour;contour.width = orderpts.size();contour.height = 1;contour.is_dense = false;contour.resize(contour.height*contour.width);for (int i = 0; i < orderpts.size(); i++){contour.points[i] = orderpts[i];}polygon.setContour(contour);viewer.addPolygon(polygon, 0, 255, 0, "ploygon", 0);while (!viewer.wasStopped()){viewer.spinOnce(1);}
}

可视化结果如下,可以发现点被有序进行排列,是有序形式

            

4、小结

   基于旋转角度进行点云排序的方法,对于图形简单的点云是有效的,比如圆、长方形、三角形等,对于复杂的,如有孔洞形式、有凹凸形式,则该方法不适用。

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

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

相关文章

Dubbo 和 Zookeeper 的关系

Dubbo 和 Zookeeper 的关系 Zookeeper的作用 zookeeper用来注册服务和进行负载均衡&#xff0c;哪一个服务由哪一个机器来提供必需让调用者知道&#xff0c;简 单来说就是ip地址和服务名称的对应关系。当然也可以通过硬编码的方式把这种对应关系在调用方 业务代码中实现&#…

AVL树讲解

AVL树 1. 概念2. AVL节点的定义3. AVL树插入3.1 旋转 4.AVL树的验证 1. 概念 AVL树是一种自平衡二叉搜索树。它的每个节点的左子树和右子树的高度差&#xff08;平衡因子&#xff0c;我们这里按右子树高度减左子树高度&#xff09;的绝对值不超过1。AVL的左子树和右子树都是AV…

MIT6.5840(6.824)Lab2总结(Raft)

MIT6.5840&#xff08;原MIT6.824&#xff09;Lab2总结&#xff08;Raft&#xff09; 资源分享&#xff1a; 官网地址&#xff1a;http://nil.csail.mit.edu/6.5840/2023/ Raft论文地址&#xff1a;http://nil.csail.mit.edu/6.5840/2023/papers/raft-extended.pdf 官方学生…

Web Servlet

目录 1 简介2 创建Servlet项目并成功发布运行3 新加Servlet步骤4 Servlet项目练习5 Servlet运行原理6 操作 HTTP Request头的方法(部分方法示例)7 操作 HTTP Response头的方法(部分方法示例)8 两种重定向(页面跳转)方法9 Cookie9.1 Cookie工作原理9.2 cookie构成9.3 Servlet 操…

axios的详细使用

目录 axios&#xff1a;现代前端开发的HTTP客户端王者 一、axios简介 二、axios的基本用法 1. 安装axios 2. 发起GET请求 3. 发起POST请求 三、axios的高级特性 1. 拦截器 2. 取消请求 3. 自动转换JSON数据 四、axios在前端开发中的应用 五、总结 axios&#xff1a…

【JS】判断是否安装了某个Chrome插件

前提 manifest.json 清单 下文均以manifest.json v3介绍。 因为Chrome官方文档中明确说明&#xff0c;v2已经弃用了。 ID 由于浏览器的安全策略&#xff0c;以下方法均在「已知扩展程序 ID」 的前提下才可实现。 获取扩展程序ID 进入扩展程序管理页&#xff0c;找到对应插…

Python基本数据类型之散列类型详解

前言&#xff1a; python的基本数据类型可以分为三类&#xff1a;数值类型、序列类型、散列类型&#xff0c;本文主要介绍散列类型。 一、散列类型 散列类型&#xff1a;内部元素无序&#xff0c;不能通过下标取值 1&#xff09;字典&#xff08;dict&#xff09;&#xff…

【DIY】电子制作创意作品:有趣的激光竖琴

在上海世博会的伊朗馆&#xff0c;我看到了一架没有琴弦的竖琴&#xff0c;那是众多参观者公认的伊朗馆里最有趣的展品&#xff01;参观者只要伸手穿过那架通体黑色的竖琴&#xff0c;音调就会被“奏响”。没有琴弦怎么奏响&#xff1f;工作人员为我们揭示了秘密——他按了一下…

Spring Boot搭建入门

Spring Boot简介 Spring Boot是对Spring进行的高度封装&#xff0c;是对Spring应用开发的高度简化版&#xff0c;是Spring技术栈的综合整合&#xff0c;是J2EE的一站式解决方案。想要精通Spring Boot的前提是需要熟悉Spring整套技术栈原理与内容。 Spring Boot的优点&#xf…

指针篇章-(冒泡排序详解)

冒泡排序 图解 tmp图解 内容图解 每次循环的次数减少 for循环详解 冒泡排序是一种简单的排序算法&#xff0c;它重复地遍历要排序的数列&#xff0c; 一次比较两个元素&#xff0c;如果它们的顺序错误就把它们交换过来。 遍历数列的工作是重复地进行直到没有再需要交换&…

Double和Float类

Double类 功能&#xff1a;实现对Double基本型数据的类包 构造方法&#xff1a; (double num) double Value()方法&#xff1a;返回对象中的double型数据。 Float类 功能&#xff1a;实现对float基本型数据的类包装。 构造方法&#xff1a; (float num) Float Value()方法…

云计算项目九:K8S安装

K8S安装 Kube-master安装 按照如下配置准备云主机 防火墙相关配置&#xff1a;禁用selinux&#xff0c;禁用swap&#xff0c;且在firewalld-*。上传kubernetes.zip 到跳板机 配置yum仓库&#xff08;跳板机&#xff09; 跳板机主机配置k8s软件源服务端 [rootjs ~]# yum -y…

设计模式-行为型模式-备忘录模式

备忘录&#xff08;Memento&#xff09;&#xff1a;在不破坏封装性的前提下&#xff0c;捕获一个对象的内部状态&#xff0c;并在该对象之外保存这个状态。这样以后就可将该对象恢复到原先保存的状态。[DP] //首先&#xff0c;我们定义Originator类&#xff0c;它有一个状态和…

C++初阶:类与对象(中篇)

目录 2. 类的六个默认成员函数2.1 构造函数2.1.1 构造函数的定义方式 2.2 析构函数2.2.1 析构函数定义方式 2.3拷贝构造函数2.3.1 拷贝构造函数的定义方式2.3.2 深拷贝与浅拷贝 2.4 赋值运算符的重载2.4.1 运算符重载2.4.2 运算符的重载的定义方式2.4.3 默认成员函数&#xff1…

HTML使用

文章目录 一、简介二、HTML快速入门三、基础标签四、图片、音频、视频标签五、超链接标签六、列表标签七、表格标签八、布局标签九、表单标签十、表单向标签 一、简介 二、HTML快速入门 ​ <html><head><title>你好</title></head><body>再…

功能强大使用简单的截图/贴图工具,PixPin

一、下载链接 PixPin 截图/贴图/长截图/文字识别/标注 | PixPin 截图/贴图/长截图/文字识别/标注 (pixpinapp.com) 二、功能 截图/贴图/长截图/文字识别/标注 三、安装教程 根据提示安装即可&#xff1a; 四、快捷键 1.软件自带快捷键&#xff08;右击PixPin查看 &#xff09…

微信小程序-入门

1.下载和安装Npm&#xff1a;Npm https://docs.npmjs.com/downloading-and-installing-node-js-and-npm 或者 https://nodejs.org/en/download/ 未安装npm 提示 以下以安装node安装包为例 按任意键继续 安装完成后 2. 下载和安装小程序开发工具 &#xff1a;https:/…

安信可IDE(AiThinker_IDE)编译ESP8266工程方法

0 工具准备 AiThinker_IDE.exe ESP8266工程源码 1 安信可IDE&#xff08;AiThinker_IDE&#xff09;编译ESP8266工程方法 1.1 解压ESP8266工程文件夹 我们这里使用的是NON-OS_SDK&#xff0c;将NON-OS_SDK中的1_UART文件夹解压到工作目录即可 我这里解压到了桌面&#xff0c…

T1 小美的数组询问(15分) - 美团编程题 题解

考试平台&#xff1a; 牛客网 题目类型&#xff1a; 30道单选题&#xff08;60分&#xff09; 2 道编程题 &#xff08;15分 25分&#xff09; 考试时间&#xff1a; 2024-03-09 &#xff08;两小时&#xff09; 题目描述 小美拿到了一个由正整数组成的数组&#xff0c;但其中…

Draco点云压缩测试

ref&#xff1a;https://github.com/google/dracohttps://codelabs.developers.google.com/codelabs/draco-3d/index.html#6 Draco Draco 是一个用于编码压缩和解压缩 3D 几何网格和点云的库&#xff0c;从而改进 3D 图形的存储和传输该代码支持压缩点、连接信息、纹理坐标、颜…