PCL点云库入门——PCL库可视化之CloudViewer类简单点云信息显示

1、前言

        可视化(visualization)涉及运用计算机图形学和图像处理技术,将数据转换成图像并在屏幕上展示,同时支持交互式处理。在PCL库中,一系列强大的可视化工具可供使用,其中较为流行的包括CloudViewer和PCLVisualizer类。CloudViewer是一个用于点云数据的简易可视化工具,而PCLVisualizer类则提供了PCL中最全面的可视化功能。相较于CloudViewer,PCLVisualizer的操作更为复杂,但其功能也更为全面。本节和下节将逐步介绍这两种工具的使用方法,从基础到高级,内容将不断更新。

2、CloudViewer类可视化

        CloudViewer以其简洁直观的设计,成为点云数据可视化的理想选择,尤其适合初学者和需要快速查看点云数据的用户。使用时,只需将点云数据加载到CloudViewer中,即可立即在屏幕上看到数据的三维图像,从而快速了解数据的整体结构和分布。下面从CloudViewer类中的主要成员函数说明到示例应用对其进行讲解说明,供大家参考学习。

2.1、主要成员函数

         CloudViewer类在#include <pcl/visualization/cloud_viewer.h>中可找到,其的成员函数不多,可见其提供的可视化功能简单,从中可知 CloudViewer类只提供如下4中点云类型的显示。

 //只包含三维点(x,y,z,)和颜色(r,g,b)+透明度(a)信息的点云显示
 using ColorACloud = pcl::PointCloud<pcl::PointXYZRGBA>;
 
//只包含三维点(x,y,z,)和颜色(r,g,b)信息的点云显示
 using ColorCloud = pcl::PointCloud<pcl::PointXYZRGB>;
 
//只包含三维点(x,y,z,)和强度(I)信息的点云显示
 using GrayCloud = pcl::PointCloud<pcl::PointXYZI>;
 
//只包含三维点(x,y,z)的点云显示
 using MonochromeCloud = pcl::PointCloud<pcl::PointXYZ>;

        主要的成员函数和变量如下表:

函数名
描述
 CloudViewer (const std::string& window_name)
类构造函数,参数为可视化窗口的名称,如PCLVis
using VizCallable = std::function<void (pcl::visualization::PCLVisualizer&)>;
可视化回调函数定义
void showCloud (const ColorCloud::ConstPtr &cloud,
 const std::stringcloudname = "cloud");
可视化化带有RGB颜色的点云数据,通过Cloudname控制显示那个点云,当有多个点云时
void showCloud (const ColorACloud::ConstPtr &cloud
const std::stringcloudname = "cloud");
可视化化带有颜色+α通道的点云数据,通过Cloudname控制显示那个点云,当有多个点云时
void showCloud (const GrayCloud::ConstPtr &cloud,
 const std::stringcloudname = "cloud");
可视化化带有强度信息的点云数据,通过Cloudname控制显示那个点云,当有多个点云时
void showCloud (const MonochromeCloud::ConstPtr &cloud,
 const std::stringcloudname = "cloud");
可视化化没有RGB颜色的点云数据,通过Cloudname控制显示那个点云,当有多个点云时
bool  wasStopped  ( int  millis_to_wait  = 1);
判断UI界面是否退出,
void  runOnVisualizationThread  ( VizCallable  x ,  const  std :: string &  key  =  "callable" ); 
在渲染输出时每次都调用接口函数 
void  runOnVisualizationThreadOnce  ( VizCallable  x );
在渲染输出执行一次调用接口函数 
void  removeVisualizationCallable  ( const  std :: string &  key  =  "callable" );
响应清除可视化界面回调函数 
inline  boost :: signals2 :: connection  registerKeyboardCallback  ( void  (* callback ) ( const  pcl :: visualization :: KeyboardEvent &,  void *),  void *  cookie  =  nullptr )
响应键盘事件回调函数
inline  boost :: signals2 :: connection  registerMouseCallback  ( void  (* callback ) ( const  pcl :: visualization :: MouseEvent &,  void *),  void *  cookie  =  nullptr )
响应鼠标事件回调函数
inline  boost :: signals2 :: connection  registerPointPickingCallback  ( void  (* callback ) ( const  pcl :: visualization :: PointPickingEvent &,  void *),  void *  cookie  =  nullptr )
响应单击选点事件回调函数

 2.2、主要函数示例

        1)、void runOnVisualizationThreadOnce (VizCallable x),该函数在渲染输出时只执行一次调用,参数为回调函数。回调函数定义为如下:

// 回调函数,在主函数中只注册一次 ,设置可视化背景颜色,添加3D文字
void viewerOneOff(pcl::visualization::PCLVisualizer& viewer)
{
    viewer.setBackgroundColor(0.0, 0.40784, 0.5451); //设置背景颜色
    pcl::PointXYZ pose;                          //添加属性的位置
    pose.x = 10.0;
    pose.y = 0;
    pose.z = 0;
   //添加PCL LOGO 3D 文字
   //参数分别为:3D文字内容,文字位置,文字大小,文字颜色
       viewer.addText3D("PCL LOGO", pose,2,1,0.647,0);
    std::cout << "Only run once" << std::endl;
}

 2)、void runOnVisualizationThread (VizCallable x, const std::string& key = "callable"),该函数在渲染输出时每次都调用,参数为回调函数。回调函数定义为如下:

// 调函数,注册后每帧显示都执行一次,可视化添加一个刷新显示字符串
void viewerPsycho(pcl::visualization::PCLVisualizer& viewer)
{
       static unsigned count = 0;
       std::stringstream ss;
       ss << "Point Cloud Library: " << count++;
       viewer.removeShape("text", 0);
       // 参数含义:刷新字符串,在屏幕位置(x,y),字体大(100),字体颜色(1,0.54,0.0),字符串id,窗口id
       viewer.addText(ss.str(), 20, 500, 40, 1, 0.54, 0, "text", 0);
}

3)、inline boost::signals2::connection registerPointPickingCallback (void (*callback) (const pcl::visualization::PointPickingEvent&, void*), void* cookie = nullptr),该函数响应单击选点事件操作,操作相对其他两个复杂,具体操作如下,先定义回调参数类型:

struct callback_args {
    pcl::PointCloud<pcl::PointXYZ>::Ptr clicked_points_3d;
    pcl::visualization::PCLVisualizer::Ptr viewerPtr;
};

        在定义回调函数如下:


void pp_callback(const pcl::visualization::PointPickingEvent& event, void* args)
{
	struct callback_args* data = (struct callback_args*)args;
	if (event.getPointIndex() == -1)
		return;
	//获取事件点的ID
	int index = event.getPointIndex();
	std::cout << "index: " << index << std::endl;
	//获取点的三维坐标
	pcl::PointXYZ current_point;
	event.getPoint(current_point.x, current_point.y, current_point.z);
	data->clicked_points_3d->points.push_back(current_point);
	//设置可视点的颜色信息
	pcl::visualization::PointCloudColorHandlerCustom<pcl::PointXYZ> red(data->clicked_points_3d, 255, 0, 0);
	data->viewerPtr->removePointCloud("clicked_points");
	//添加可视点
	data->viewerPtr->addPointCloud(data->clicked_points_3d, red, "clicked_points");
	//设置可视点的大小
	data->viewerPtr->setPointCloudRenderingProperties(pcl::visualization::PCL_VISUALIZER_POINT_SIZE, 10, "clicked_points");
	//输出点的信息
	std::cout << current_point.x << " " << current_point.y << " " << current_point.z << std::endl;
}

注意:以上3的)内容我们会在下节进行结果展示,大家可以先跳过内容。 

3、CloudViewer可视化点云信息Demo

3.1、示例1,可视化无颜色点云和有颜色点云

        示例代码如下:

/*****************************************************************//**
* \file   PCLviewerMian.cpp
* \brief
*
* \author YZS
* \date   December 2024
*********************************************************************/
#include <iostream>              
#include <string>
#include <pcl/io/pcd_io.h>
#include <pcl/visualization/cloud_viewer.h>  //cloud_viewer类的头文件
using namespace std;

// 回调函数,在主函数中只注册一次 ,设置可视化背景颜色,添加3D文字
void viewerOneOff(pcl::visualization::PCLVisualizer& viewer)
{
	viewer.setBackgroundColor(0.0, 0.40784, 0.5451); //设置背景颜色
	pcl::PointXYZ pose;                          //添加属性的位置
	pose.x = 0.10;
	pose.y = 0;
	pose.z = 0;
	//添加PCL LOGO 3D 文字
	//参数分别为:3D文字内容,文字位置,文字大小,文字颜色
	viewer.addText3D("PCL LOGO", pose, 0.5, 1, 0.647, 0);
	std::cout << "Only run once" << std::endl;
}
// 调函数,注册后每帧显示都执行一次,可视化添加一个刷新显示字符串
void viewerPsycho(pcl::visualization::PCLVisualizer& viewer)
{
	static unsigned count = 0;
	std::stringstream ss;
	ss << "Point Cloud Library: " << count++;
	viewer.removeShape("text", 0);
	// 参数含义:刷新字符串,在屏幕位置(x,y),字体大(100),字体颜色(1,0.54,0.0),字符串id,窗口id
	viewer.addText(ss.str(), 20, 500, 50, 1, 0.54, 0, "text", 0);
}
// 可视化点为PointXYZ的点云
void TestVisPointClouXYZ()
{
	// 创建点云对象,显示点类型为(X,Y,Z)的点云
	pcl::PointCloud<pcl::PointXYZ>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZ>);
	//读取点云文件
	std::string fileName = "E:/PCLlearnData/4/fragment.pcd";
	pcl::io::loadPCDFile(fileName, *cloud);
	//创建可视化句柄viewer对象
	pcl::visualization::CloudViewer viewer("PCLVisPointXYZ");
	//用showCloud,将Cloud可视化到屏幕
	viewer.showCloud(cloud);

	//该接口函数在可视化的时候只执行一次,可以用于设置背景等一次性操作
	viewer.runOnVisualizationThreadOnce(viewerOneOff);
	//该接口函数在渲染输出时每次都调用,可以用于设置显示时间戳、计时器和动态处理等操作
	viewer.runOnVisualizationThread(viewerPsycho);
	while (!viewer.wasStopped())
	{
		//此处可以添加其他处理
	}
}
//可视化点为PointXYZRGB的点云数据
void TestVisPointClouXYZRGB()
{
	// 创建点云对象,显示点类型为(X,Y,Z ,R,G,B)的点云
	pcl::PointCloud<pcl::PointXYZRGB>::Ptr cloud(new  pcl::PointCloud<pcl::PointXYZRGB>);
	//读取点云文件
	std::string fileName = "E:/PCLlearnData/4/fragment.pcd";
	pcl::io::loadPCDFile(fileName, *cloud);
	//创建可视化句柄viewer对象
	pcl::visualization::CloudViewer viewer("PCLVisPointXYZRGB");
	//用showCloud,将Cloud可视化到屏幕
	viewer.showCloud(cloud);
	while (!viewer.wasStopped())
	{
		//此处可以添加其他处理
	}
}


int main(int argc, char* argv[])
{
	//TestVisPointClouXYZ();
	TestVisPointClouXYZRGB();
	std::cout << "Hello PCL!" << std::endl;
	std::system("pause");
	return 0;
}

        结果展示:

3.2、键盘操作展示

        在上面2.1节内容中提到键盘和鼠标回调操作,常用的键盘操作命令有如下:

键盘操作命令

说明

p

切换到基于点的表示

w

切换到基于线框的表示

s

切换到基于表面的表示

c

显示当前摄像头/窗口参数

f

飞到点模式

e

退出交互程序

q

停止并调用VTK的TerminateApp

o

在透视/平行投影之间切换

g

显示比例栅格(开/关)

u

显示查找表(开/关)

        1)、显示比例栅格展示如下,其他的命令大家可自行操作一下查看效果。

至此完成第四节PCL库可视化之CloudViewer类简单点云信息显示学习,下一节我们将进入《PCL库可视化之PCLVisualizer类复杂点云信息显示》的学习。 

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

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

相关文章

人工智能技术的深度解析与推广【人工智能的应用场景】

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c; 忍不住分享一下给大家。点击跳转到网站 学习总结 1、掌握 JAVA入门到进阶知识(持续写作中……&#xff09; 2、学会Oracle数据库入门到入土用法(创作中……&#xff09; 3、手把…

Source Insight的使用经验汇总

01-Add All"和“Add Tree”有何区别&#xff1f; 在 Source Insight 中&#xff0c;“Add All”和“Add Tree”是两种向项目&#xff08;Project&#xff09;中添加文件的操作选项&#xff0c;它们的区别在于处理文件和目录的方式不同&#xff1a; 1. Add All 范围&am…

检查读取数据寄存器输出的多扇出

为使第二寄存器被 RAM 原语吸收&#xff0c;来自存储器阵列的数据输出位的扇出必须为 1 。这在下图中进行了说明。 检查地址 / 读取数据寄存器上的复位信号 不应复位存储器阵列。只有 RAM 的输出可以容许复位。复位必须是同步的&#xff0c;以便将输出寄存器推断到 RAM 基元…

信创改造-Spring Boot 项目部署至 TongWeb

打 war 包参考&#xff1a;https://blog.csdn.net/z1353095373/article/details/144330999

趣味编程:猜拳小游戏

1.简介 这个系列的第一篇以猜拳小游戏开始&#xff0c;这是源于我们生活的灵感&#xff0c;在忙碌的时代中&#xff0c;我们每个人都在为自己的生活各自忙碌着&#xff0c;奔赴着自己所走向的那条路上&#xff0c;即使遍体鳞伤。 但是&#xff0c;生活虽然很苦&#xff0c;也不…

微服务-01

1.认识微服务 1.1 单体架构 单体架构&#xff08;monolithic structure&#xff09;&#xff1a;顾名思义&#xff0c;整个项目中所有功能模块都在一个工程中开发&#xff1b;项目部署时需要对所有模块一起编译、打包&#xff1b;项目的架构设计、开发模式都非常简单。 当项目…

【Linux———基础IO】

每一滴眼泪&#xff0c;每一次心碎&#xff0c;什么爱能无疚无悔.......................................................................... 文章目录 前言 一、【认识文件I/O】 1.1、【回顾C语言文件I/O】 1.2、【操作系统文件I/O】 1.2.1、【open函数】 1、【open函数的三…

机器视觉LED面光源 成像效果显著

在机器视觉系统中&#xff0c;光源作为不可或缺的一部分&#xff0c;合适的光源能够提高检测物成像效果&#xff0c;增强检测效果。因此光源的选择至关重要&#xff0c;针对不同的检测对象,不同形状的光源应运而生。每种光源都有其适用的场景应用&#xff0c;选对光源&#xff…

HBase分布式安装配置(Zookeeper+HBase)

HBase 是一个面向列式存储的分布式数据库&#xff0c;其设计思想来源于 Google 的 BigTable 论文。HBase 底层存储基于 HDFS 实现&#xff0c;集群的管理基于 ZooKeeper 实现。HBase 良好的分布式架构设计为海量数据的快速存储、随机访问提供了可能&#xff0c;基于数据副本机制…

【2024 Dec 超实时】编辑安装llama.cpp并运行llama

首先讲一下环境 这是2024 年12月&#xff0c;llama.cpp 的编译需要cmake 呜呜呜 网上教程都是make 跑的。反正我现在装的时候make已经不再适用了&#xff0c;因为工具的版本&#xff0c;捣鼓了很久。 ubuntu 18 conda env内置安装。 以下是可以完美编译llama.cpp的测试工具版…

go-zero(十二)消息队列

go zero 消息队列 在微服务架构中&#xff0c;消息队列主要通过异步通信实现服务间的解耦&#xff0c;使得各个服务可以独立发展和扩展。 go-zero中使用的队列组件go-queue&#xff0c;是gozero官方实现的基于Kafka和Beanstalkd 的消息队列框架,我们使用kafka作为演示。 一、…

vue绕过rules自定义编写动态校验

今天犯了个低级错误&#xff0c;虽然走了很多弯路&#xff0c;但这个过程还是值得记录一下 例子如下&#xff0c;有两个输入框&#xff1a; 第一个是套餐选择下拉框&#xff0c;可以下拉选择三个内容 第二个要根据上面的套餐选择三个选项来决定怎么显示&#xff0c;使用v-if&…

ABAQUS进行焊接仿真分析(含子程序)

0 前言 焊接技术作为现代制造业中的重要连接工艺,广泛应用于汽车、船舶、航空航天、能源等多个行业。焊接接头的质量和性能直接影响到结构件的安全性、可靠性和使用寿命。因此,在焊接过程中如何有效预测和优化焊接过程中的热效应、应力变化以及材料变形等问题,成为了焊接研…

【efinance一个2k星的库】

efinance 是一个可以快速获取基金、股票、债券、期货数据的 Python 库&#xff0c;回测以及量化交易的好帮手 但没有等比复权的&#xff0c;不用。 import efinance as ef ef.stock.get_quote_history(510880,fqt2)

【考前预习】3.计算机网络—数据链路层

往期推荐 【考前预习】2.计算机网络—物理层-CSDN博客 【考前预习】1.计算机网络概述-CSDN博客 浅谈云原生--微服务、CICD、Serverless、服务网格_云原生cicd-CSDN博客 子网掩码、网络地址、广播地址、子网划分及计算_子网广播地址-CSDN博客 浅学React和JSX-CSDN博客 目录 1.数…

Microsemi Libero SoC免费许可证申请指南(Microchip官网2024最新方法)

点击如下链接&#xff1a; https://www.microchip.com/en-us/products/fpgas-and-plds/fpga-and-soc-design-tools/fpga/licensing 点击右侧&#xff0c;请求免费的License 如果提示登录&#xff0c;请先登录Microchip账号。 点击Request Free License。 选项一年免费的Li…

嵌入式Linux应用层开发——调试专篇(关于使用GDB调试远程下位机开发板的应用层程序办法 + VSCode更好的界面调试体验提升)

环境预备——调试 虽说有正点原子的代码带着&#xff0c;但是&#xff0c;如果我们只是打着printf这种方式进行手动的检查代码错误&#xff0c;还是不太方便的&#xff0c;笔者这里整理了两个上位机调试路线。 路线1&#xff1a;使用GCC7.5&#xff0c;这个路线比较保守&#…

深度学习训练参数之学习率介绍

学习率 1. 什么是学习率 学习率是训练神经网络的重要超参数之一&#xff0c;它代表在每一次迭代中梯度向损失函数最优解移动的步长&#xff0c;通常用 η \eta η 表示。它的大小决定网络学习速度的快慢。在网络训练过程中&#xff0c;模型通过样本数据给出预测值&#xff0…

蒙特卡洛模拟(Monte Carlo Simulation)详解

简介&#xff1a;个人学习分享&#xff0c;如有错误&#xff0c;欢迎批评指正。 历史背景 蒙特卡洛模拟的名称来源于摩纳哥的蒙特卡洛赌场&#xff0c;因其依赖于随机性和概率&#xff0c;与赌博中的随机过程有相似之处。该方法的雏形可以追溯到20世纪40年代&#xff0c;二战期…

Git-分支(branch)常用命令

分支 我们在做项目开发的时候&#xff0c;无论是软件项目还是其他机械工程项目&#xff0c;我们为了提高效率以及合理的节省时间等等原因&#xff0c;现在都不再是线性进行&#xff0c;而是将一个项目抽离出诸进行线&#xff0c;每一条线在git中我们就叫做分支&#xff0c;bran…