PCL提取3D点云模型特征(1.0 点云曲率)

本文介绍如何使用PCL库计算点云的曲率并进行特征提取,详细讲解了NormalEstimation类的使用方法,包括设置输入点云、初始化邻域搜索、设置搜索方法等步骤。并通过代码示例展示了如何根据曲率值对点云进行着色,最后保存为PLY文件。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、随便扯扯的概述

        在进入到计算机图形学的研究中已经过去了好几个月了,自然免不了要跟PCL打交道。在学习PCL的过程中,越来越觉得PCL真的是个非常强大的工具,让人爱不释手,但同时也让人感到沮丧,因为你会发现,你有的想去实现的想法PCL里面都早就实现了,并且效果还非常好。这里就我在学习提取点云特征的过程中遇到的一个PCL里面的一个非常简单并且基本的特征提取方法(利用PCL中的曲率计算)做个简要的记录。

         如果需要配置PCL,请看我https://blog.youkuaiyun.com/McQueen_LT/article/details/84792197这篇文章。

 

二、计算点云曲率(法线)的具体步骤

        PCL中有一个计算点云法线的类叫做NormalEstimation,使用方法就是:

        1. 利用setInputCloud()方法输入需要进行法线估计的点云;

        2. 初始化一个邻域搜索的方式,例如KdTree;

        3. 使用setSearchMethod()设置搜索方式;

        4. 声明一个用来存储计算结果的Normal类点云;

        5. 通过setRadiusSearch()设置搜索邻域的半径;

        6. 将第4步声明的变量传入compute()函数存储计算结果。

 

三、了解 pcl::PointCloud<pcl::Normal>::Ptr 数据结构

       在二中最终计算的结果就存储在这个类型的数据结构中,在后面的利用曲率提取特征的过程中我们会利用到这个数据结构中的curvature元素。要找到这个元素也不是特别明显就能发现的,可以通过阅读PCL的相关源码http://docs.pointclouds.org/trunk/point__types_8hpp_source.html就能很清楚其中的内部结构。

       这个结构中一共有四个元素,前三个元素分别为normal_x,normal_y,normal_z,可以清楚地知道分别就是法线的向量了,还有一个就是curvature元素,表示当前点的曲率。

 

四、 提取特征(代码)

       后面的过程更简单,我就直接贴出完整的代码。

#include <pcl/point_types.h>
#include <pcl/features/pfh.h>
#include <pcl/features/fpfh.h>
#include <pcl/io/ply_io.h>
#include <pcl/io/pcd_io.h>
#include <pcl/features/normal_3d.h>
#include <pcl/visualization/histogram_visualizer.h>
#include <pcl/visualization/pcl_plotter.h>
#include <pcl/visualization/cloud_viewer.h>
#include <math.h>
#include <Eigen/Eigen>
#include <cmath>
#include <string>
using namespace std;

int main()
{
    string filename="bunny";
    pcl::PointCloud<pcl::PointXYZ>::Ptr cloud_ptr (new pcl::PointCloud<pcl::PointXYZ>);

    pcl::PLYReader reader;
    //    pcl::PCDReader reader;

    //    reader.read(filename+".pcd", *cloud_ptr);
    reader.read(filename+".ply", *cloud_ptr);
    if(cloud_ptr==NULL)
    {
        cout<<"ply/pcd file read error"<<endl;
        return -1;
    }


    pcl::NormalEstimation<pcl::PointXYZ, pcl::Normal> ne;
    ne.setInputCloud(cloud_ptr);
    pcl::search::KdTree<pcl::PointXYZ>::Ptr tree (new pcl::search::KdTree<pcl::PointXYZ>());
    ne.setSearchMethod(tree);
    pcl::PointCloud<pcl::Normal>::Ptr cloud_normals (new pcl::PointCloud<pcl::Normal>);
    double nor_radius=0.002;
    
    ne.setRadiusSearch(nor_radius);
    ne.compute(*cloud_normals);

    pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud_b(new pcl::PointCloud<pcl::PointXYZRGB>);
    cloud_b->points.resize(cloud_ptr->size());

    for(int i=0; i<cloud_ptr->points.size(); i++)
    {
        cloud_b->points[i].x = cloud_ptr->points[i].x;
        cloud_b->points[i].y = cloud_ptr->points[i].y;
        cloud_b->points[i].z = cloud_ptr->points[i].z;
        cloud_b->at(i).r=100;
        cloud_b->at(i).g=100;
        cloud_b->at(i).b=100;
    }

    pcl::KdTreeFLANN<pcl::PointXYZRGB> kdtree;
    kdtree.setInputCloud(cloud_b);

    for(int i=0; i<cloud_b->points.size(); i++)
    {
        if(cloud_normals->points[i].curvature>0.01)
        {
            cloud_b->at(i).r=255;
            cloud_b->at(i).g=0;
            cloud_b->at(i).b=0;
        }
    }

    pcl::io::savePLYFile<pcl::PointXYZRGB>("curvature_"+filename+".ply", *cloud_b, false);
    pcl::visualization::CloudViewer viewer("Viewer");
    viewer.showCloud(cloud_b);
    system("PAUSE");

    return 0;
}

最后贴两张效果图(上述代码会将最后的结果以PLY文件的格式保存到本地,如果想查看的话去去下载个meshlab就行.)

 

 

### PCL库中提取有线轨道的方法 通过PCL(Point Cloud Library),可以利用多种算法来处理点云数据并从中提取特定结构,比如有线轨道。以下是基于提供的引用内容以及专业知识的一种实现方法。 #### 法线估计与区域生长聚类 为了从复杂的点云场景中分离出目标对象(如电线或轨道),可以通过法线估计和区域生长算法完成初步的分割操作。这种方法的核心在于先估算每个点的局部几何特性(即法向量方向),再依据这些特性和空间邻接关系构建连通组件[^1]。 ```cpp #include <pcl/point_types.h> #include <pcl/features/normal_3d.h> // 创建输入点云指针 pcl::PointCloud<pcl::PointXYZ>::Ptr cloud (new pcl::PointCloud<pcl::PointXYZ>); // 假设已加载点云至变量 *cloud* // 初始化KDTrees用于加速最近邻居搜索 pcl::search::KdTree<pcl::PointXYZ>::Ptr tree (new pcl::search::KdTree<pcl::PointXYZ>); tree->setInputCloud(cloud); // 定义存储法线的方向容器 pcl::PointCloud<pcl::Normal>::Ptr normals(new pcl::PointCloud<pcl::Normal>()); // 设置法线估计器参数 pcl::NormalEstimation<pcl::PointXYZ, pcl::Normal> ne; ne.setInputCloud(cloud); ne.setSearchMethod(tree); ne.setKSearch(20); // 考虑周围20个临近点 ne.compute(*normals); ``` 上述代码片段展示了如何使用`pcl::NormalEstimation`模块计算点云上每一点的法线信息[^3]。此步骤有助于区分不同类型的表面特征(例如平坦面和平滑曲线)。 接着,在获得法线的基础上执行区域增长策略进一步细化分类: ```cpp #include <pcl/segmentation/region_growing.h> // 配置RegionGrowing实例 pcl::RegionGrowing<pcl::PointXYZ, pcl::Normal> rg; rg.setMinClusterSize(50); // 最少包含多少点才视为有效簇群 rg.setMaxClusterSize(1000000); // 单一簇的最大规模限制 rg.setNumberOfNeighbours(30); // 寻找近邻数量设定为30 rg.setSmoothnessThreshold(3.0 / 180.0 * M_PI); // 平滑角度阈值设置成3度以内变化允许范围 rg.setCurvatureThreshold(1.0); // 曲率差异容忍限度定义 // 输入原始点集及其对应法线结果给算法模型 rg.setInputCloud(cloud); rg.setInputNormals(normals); std::vector<pcl::PointIndices> clusters; // 存储最终得到的所有独立子集合索引列表 rg.extract(clusters); ``` 这里采用的是`pcl::RegionGrowing`工具来进行更精细层次上的分组作业。它会按照指定条件自动把相近属性的一系列节点组合起来构成新的整体单元——也就是所谓的“cluster”。 #### PCA降维分析 当面对更加复杂的情况时,则可引入主成分分析技术(PCA),帮助识别潜在的主要趋势轴线,并据此判断哪些部分最有可能代表所寻找的目标实体—像电缆那样细长延伸的对象往往会在某个维度占据主导地位[^2]: ```cpp #include <Eigen/Dense> #include <pcl/common/eigen.h> for(auto &indices : clusters){ Eigen::Vector4f centroid; pcl::compute3DCentroid(*cloud(indices),centroid); Eigen::Matrix3f covariance_matrix = Eigen::Matrix3f::Zero(); pcl::computeCovarianceMatrixNormalized(*cloud(indices), centroid, covariance_matrix ); Eigen::SelfAdjointEigenSolver<Eigen::Matrix3f> eigen_solver(covariance_matrix,Eigen::ComputeEigenvectors ); float lambda_x=eigen_solver.eigenvalues()[2]; float lambda_y=eigen_solver.eigenvalues()[1]; float lambda_z=eigen_solver.eigenvalues()[0]; double linearity=(lambda_x-lambda_y)/lambda_x; if(linearity > threshold){ /*threshold is user defined*/ } // 认定当前团块可能是条状物... } ``` 以上程序段落说明了怎样借助协方差矩阵分解获取各个方向的重要性权重比例,从而定量评估某一群体是否具备显著延展性的特质。 最后一步便是筛选满足既定标准的结果作为候选解输出。 --- ### 性能优化建议 考虑到实际工程项目里可能会遇到海量级的数据样本,单纯依靠逐帧扫描的方式效率低下难以接受;此时不妨考虑借鉴文献提到过的积分图像快速逼近方案替代传统最小二乘拟合法求导过程加快运算速度的同时保持足够的精度水平。 此外还有其他高级技巧可供参考学习,例如结合机器视觉领域流行的深度神经网络框架训练专门针对某种特殊形态物体检测的任务模型等等[^4]。 ---
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值