博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
PCL系列——怎样逐渐地配准一对点云
阅读量:6504 次
发布时间:2019-06-24

本文共 10076 字,大约阅读时间需要 33 分钟。

欢迎訪问

PCL系列

说明

通过本教程,我们将会学会:

  • 怎样配准多个点云图。

  • 配准的方法是:点云图两两配准,计算它们的变换矩阵。然后计算总的变换矩阵。
  • 两个点云配准使用的是非线性ICP算法,它是ICP的算法的变体,使用Levenberg-Marquardt最优化。

操作

  • 在VS2010 中新建一个文件 pairwise_incremental_registration.cpp,然后将以下的代码拷贝到文件里。
  • 參照之前的文章。配置项目的属性。设置包括文件夹和库文件夹和附加依赖项。

#include 
//共享指针//点/点云#include
#include
#include
//pcd文件输入/输出#include
//滤波#include
#include
//特征#include
//配准#include
//ICP方法#include
#include
//可视化#include
//命名空间using pcl::visualization::PointCloudColorHandlerGenericField;using pcl::visualization::PointCloudColorHandlerCustom;//定义类型的别名typedef pcl::PointXYZ PointT;typedef pcl::PointCloud
PointCloud;typedef pcl::PointNormal PointNormalT;typedef pcl::PointCloud
PointCloudWithNormals;//全局变量//可视化对象pcl::visualization::PCLVisualizer *p;//左视区和右视区,可视化窗体分成左右两部分int vp_1, vp_2;//定义结构体,用于处理点云struct PCD{ PointCloud::Ptr cloud; //点云指针 std::string f_name; //文件名称 //构造函数 PCD() : cloud (new PointCloud) {}; //初始化};// 定义新的点表达方式< x, y, z, curvature > 坐标+曲率class MyPointRepresentation : public pcl::PointRepresentation
//继承关系{ using pcl::PointRepresentation
::nr_dimensions_;public: MyPointRepresentation () { //指定维数 nr_dimensions_ = 4; } //重载函数copyToFloatArray,以定义自己的特征向量 virtual void copyToFloatArray (const PointNormalT &p, float * out) const { //< x, y, z, curvature > 坐标xyz和曲率 out[0] = p.x; out[1] = p.y; out[2] = p.z; out[3] = p.curvature; }};//在窗体的左视区。简单的显示源点云和目标点云void showCloudsLeft(const PointCloud::Ptr cloud_target, const PointCloud::Ptr cloud_source){ p->removePointCloud ("vp1_target"); //依据给定的ID。从屏幕中去除一个点云。參数是ID p->removePointCloud ("vp1_source"); // PointCloudColorHandlerCustom
tgt_h (cloud_target, 0, 255, 0); //目标点云绿色 PointCloudColorHandlerCustom
src_h (cloud_source, 255, 0, 0); //源点云红色 p->addPointCloud (cloud_target, tgt_h, "vp1_target", vp_1); //载入点云 p->addPointCloud (cloud_source, src_h, "vp1_source", vp_1); PCL_INFO ("Press q to begin the registration.\n"); //在命令行中显示提示信息 p-> spin();}//在窗体的右视区,简单的显示源点云和目标点云void showCloudsRight(const PointCloudWithNormals::Ptr cloud_target, const PointCloudWithNormals::Ptr cloud_source){ p->removePointCloud ("source"); //依据给定的ID。从屏幕中去除一个点云。

參数是ID

p->removePointCloud ("target"); PointCloudColorHandlerGenericField<PointNormalT> tgt_color_handler (cloud_target, "curvature"); //目标点云彩色句柄 if (!tgt_color_handler.isCapable ()) PCL_WARN ("Cannot create curvature color handler!"); PointCloudColorHandlerGenericField<PointNormalT> src_color_handler (cloud_source, "curvature"); //源点云彩色句柄 if (!src_color_handler.isCapable ()) PCL_WARN ("Cannot create curvature color handler!"); p->addPointCloud (cloud_target, tgt_color_handler, "target", vp_2); //载入点云 p->addPointCloud (cloud_source, src_color_handler, "source", vp_2); p->spinOnce(); } // 读取一系列的PCD文件(希望配准的点云文件) // 參数argc 參数的数量(来自main()) // 參数argv 參数的列表(来自main()) // 參数models 点云数据集的结果向量 void loadData (int argc, char **argv, std::vector<PCD, Eigen::aligned_allocator<PCD> > &models) { std::string extension (".pcd"); //声明并初始化string类型变量extension,表示文件后缀名 // 通过遍历文件名称,读取pcd文件 for (int i = 1; i < argc; i++) //遍历全部的文件名称(略过程序名) { std::string fname = std::string (argv[i]); if (fname.size () <= extension.size ()) //文件名称的长度是否符合要求 continue; std::transform (fname.begin (), fname.end (), fname.begin (), (int(*)(int))tolower); //将某操作(小写字母化)应用于指定范围的每一个元素 //检查文件是否是pcd文件 if (fname.compare (fname.size () - extension.size (), extension.size (), extension) == 0) { // 读取点云,并保存到models PCD m; m.f_name = argv[i]; pcl::io::loadPCDFile (argv[i], *m.cloud); //读取点云数据 //去除点云中的NaN点(xyz都是NaN) std::vector<int> indices; //保存去除的点的索引 pcl::removeNaNFromPointCloud(*m.cloud,*m.cloud, indices); //去除点云中的NaN点 models.push_back (m); } } } //简单地配准一对点云数据,并返回结果 //參数cloud_src 源点云 //參数cloud_tgt 目标点云 //參数output 输出点云 //參数final_transform 成对变换矩阵 //參数downsample 是否下採样 void pairAlign (const PointCloud::Ptr cloud_src, const PointCloud::Ptr cloud_tgt, PointCloud::Ptr output, Eigen::Matrix4f &final_transform, bool downsample = false) { // //为了一致性和速度,下採样 // \note enable this for large datasets PointCloud::Ptr src (new PointCloud); //创建点云指针 PointCloud::Ptr tgt (new PointCloud); pcl::VoxelGrid<PointT> grid; //VoxelGrid 把一个给定的点云,聚集在一个局部的3D网格上,并下採样和滤波点云数据 if (downsample) //下採样 { grid.setLeafSize (0.05, 0.05, 0.05); //设置体元网格的叶子大小 //下採样 源点云 grid.setInputCloud (cloud_src); //设置输入点云 grid.filter (*src); //下採样和滤波,并存储在src中 //下採样 目标点云 grid.setInputCloud (cloud_tgt); grid.filter (*tgt); } else //不下採样 { src = cloud_src; //直接复制 tgt = cloud_tgt; } //计算曲面的法向量和曲率 PointCloudWithNormals::Ptr points_with_normals_src (new PointCloudWithNormals); //创建源点云指针(注意点的类型包括坐标和法向量) PointCloudWithNormals::Ptr points_with_normals_tgt (new PointCloudWithNormals); //创建目标点云指针(注意点的类型包括坐标和法向量) pcl::NormalEstimation<PointT, PointNormalT> norm_est; //该对象用于计算法向量 pcl::search::KdTree<pcl::PointXYZ>::Ptr tree (new pcl::search::KdTree<pcl::PointXYZ> ()); //创建kd树,用于计算法向量的搜索方法 norm_est.setSearchMethod (tree); //设置搜索方法 norm_est.setKSearch (30); //设置近期邻的数量 norm_est.setInputCloud (src); //设置输入云 norm_est.compute (*points_with_normals_src); //计算法向量,并存储在points_with_normals_src pcl::copyPointCloud (*src, *points_with_normals_src); //复制点云(坐标)到points_with_normals_src(包括坐标和法向量) norm_est.setInputCloud (tgt); //这3行计算目标点云的法向量,同上 norm_est.compute (*points_with_normals_tgt); pcl::copyPointCloud (*tgt, *points_with_normals_tgt); //创建一个 自己定义点表达方式的 实例 MyPointRepresentation point_representation; //加权曲率维度。以和坐标xyz保持平衡 float alpha[4] = {
1.0, 1.0, 1.0, 1.0}; point_representation.setRescaleValues (alpha); //设置缩放值(向量化点时使用) //创建非线性ICP对象 并设置參数 pcl::IterativeClosestPointNonLinear<PointNormalT, PointNormalT> reg; //创建非线性ICP对象(ICP变体,使用Levenberg-Marquardt最优化) reg.setTransformationEpsilon (1e-6); //设置容许的最大误差(迭代最优化) //***** 注意:依据自己数据库的大小调节该參数 reg.setMaxCorrespondenceDistance (0.1); //设置相应点之间的最大距离(0.1m),在配准过程中,忽略大于该阈值的点 reg.setPointRepresentation (boost::make_shared<const MyPointRepresentation> (point_representation)); //设置点表达 //设置源点云和目标点云 //reg.setInputSource (points_with_normals_src); //版本号不符合。使用以下的语句 reg.setInputCloud (points_with_normals_src); //设置输入点云(待变换的点云) reg.setInputTarget (points_with_normals_tgt); //设置目标点云 reg.setMaximumIterations (2); //设置内部优化的迭代次数 // Run the same optimization in a loop and visualize the results Eigen::Matrix4f Ti = Eigen::Matrix4f::Identity (), prev, targetToSource; PointCloudWithNormals::Ptr reg_result = points_with_normals_src; //用于存储结果(坐标+法向量) for (int i = 0; i < 30; ++i) //迭代 { PCL_INFO ("Iteration Nr. %d.\n", i); //命令行显示迭代的次数 //保存点云,用于可视化 points_with_normals_src = reg_result; // //预计 //reg.setInputSource (points_with_normals_src); reg.setInputCloud (points_with_normals_src); //又一次设置输入点云(待变换的点云),由于经过上一次迭代,已经发生变换了 reg.align (*reg_result); //对齐(配准)两个点云 Ti = reg.getFinalTransformation () * Ti; //累积(每次迭代的)变换矩阵 //假设这次变换和上次变换的误差比阈值小。通过减小最大的相应点距离的方法来进一步细化 if (fabs ((reg.getLastIncrementalTransformation () - prev).sum ()) < reg.getTransformationEpsilon ()) reg.setMaxCorrespondenceDistance (reg.getMaxCorrespondenceDistance () - 0.001); //减小相应点之间的最大距离(上面设置过) prev = reg.getLastIncrementalTransformation (); //上一次变换的误差 //显示当前配准状态。在窗体的右视区,简单的显示源点云和目标点云 showCloudsRight(points_with_normals_tgt, points_with_normals_src); } targetToSource = Ti.inverse(); //计算从目标点云到源点云的变换矩阵 pcl::transformPointCloud (*cloud_tgt, *output, targetToSource); //将目标点云 变换回到 源点云帧 p->removePointCloud ("source"); //依据给定的ID,从屏幕中去除一个点云。參数是ID p->removePointCloud ("target"); PointCloudColorHandlerCustom<PointT> cloud_tgt_h (output, 0, 255, 0); //设置点云显示颜色。下同 PointCloudColorHandlerCustom<PointT> cloud_src_h (cloud_src, 255, 0, 0); p->addPointCloud (output, cloud_tgt_h, "target", vp_2); //加入点云数据。下同 p->addPointCloud (cloud_src, cloud_src_h, "source", vp_2); PCL_INFO ("Press q to continue the registration.\n"); p->spin (); p->removePointCloud ("source"); p->removePointCloud ("target"); //add the source to the transformed target *output += *cloud_src; // 拼接点云图(的点)点数数目是两个点云的点数和 final_transform = targetToSource; //终于的变换。目标点云到源点云的变换矩阵 } //**************** 入口函数 ************************ //主函数 int main (int argc, char** argv) { //读取数据 std::vector<PCD, Eigen::aligned_allocator<PCD> > data; //模型 loadData (argc, argv, data); //读取pcd文件数据,定义见上面 //检查用户数据 if (data.empty ()) { PCL_ERROR ("Syntax is: %s <source.pcd> <target.pcd> [*]", argv[0]); //语法 PCL_ERROR ("[*] - multiple files can be added. The registration results of (i, i+1) will be registered against (i+2), etc"); //能够使用多个文件 return (-1); } PCL_INFO ("Loaded %d datasets.", (int)data.size ()); //显示读取了多少个点云文件 //创建一个 PCLVisualizer 对象 p = new pcl::visualization::PCLVisualizer (argc, argv, "Pairwise Incremental Registration example"); //p是全局变量 p->createViewPort (0.0, 0, 0.5, 1.0, vp_1); //创建左视区 p->createViewPort (0.5, 0, 1.0, 1.0, vp_2); //创建右视区 //创建点云指针和变换矩阵 PointCloud::Ptr result (new PointCloud), source, target; //创建3个点云指针,分别用于结果。源点云和目标点云 //全局变换矩阵,单位矩阵。成对变换 //逗号表达式,先创建一个单位矩阵,然后将成对变换 赋给 全局变换 Eigen::Matrix4f GlobalTransform = Eigen::Matrix4f::Identity (), pairTransform; //遍历全部的点云文件 for (size_t i = 1; i < data.size (); ++i) { source = data[i-1].cloud; //源点云 target = data[i].cloud; //目标点云 showCloudsLeft(source, target); //在左视区。简单的显示源点云和目标点云 PointCloud::Ptr temp (new PointCloud); //创建暂时点云指针 //显示正在配准的点云文件名称和各自的点数 PCL_INFO ("Aligning %s (%d points) with %s (%d points).\n", data[i-1].f_name.c_str (), source->points.size (), data[i].f_name.c_str (), target->points.size ()); //******************************************************** //配准2个点云。函数定义见上面 pairAlign (source, target, temp, pairTransform, true); //将当前的一对点云数据,变换到全局变换中。 pcl::transformPointCloud (*temp, *result, GlobalTransform); //更新全局变换 GlobalTransform = GlobalTransform * pairTransform; //******************************************************** // 保存成对的配准结果。变换到第一个点云帧 std::stringstream ss; //这两句是生成文件名称 ss << i << ".pcd"; pcl::io::savePCDFile (ss.str (), *result, true); //保存成对的配准结果 } }
  • 又一次生成项目。
  • 到改项目的Debug文件夹下。按住Shift。同一时候点击鼠标右键,在当前窗体打开CMD窗体。
  • 在命令行中输入pairwise_incremental_registration.exe capture0001.pcd capture0002.pcd capture0003.pcd capture0004.pcd capture0005.pcd,运行程序。得到例如以下图所看到的的结果。
    结果图

參考

你可能感兴趣的文章
php session作用,session的作用是什么
查看>>
php 两行数据合并,PHP如何实现统计数据合并
查看>>
java国外研究综述,国内外研究现状_毕业论文
查看>>
php执行事务,thinkPHP框架中执行事务的方法示例
查看>>
php 两个值比较大小写,php比较两个字符串(大小写敏感)的函数strcmp()
查看>>
java3d获取模型在xoy平面的投影,将3D世界点投影到新视图图像平面
查看>>
php 修改图片src,jq修改src图片地址 .attr is not a function
查看>>
sudo php 无法打开,从PHP / Apache,exec()或system()程序作为root用户:“ sudo:无法打开审核系统:权限被拒绝”...
查看>>
sql数据库java连接sqlserver2005数据库
查看>>
clientapivc api TCP&UDP—helloworld
查看>>
下划线的学习1
查看>>
在struts2.3.4.1中使用注解、反射、拦截器实现基于方法的权限控制
查看>>
单例模式 - 程序实现(Java)
查看>>
如何隐藏Cognos Viewer
查看>>
响应式网页设计:rem、em设置网页字体大小自适应
查看>>
ImageView的属性android:scaleType
查看>>
商业智能给数据获取带来的局部效益案例
查看>>
巧妙运用二进制验证权限
查看>>
C#中的WebBrowser控件的使用
查看>>
(转)<Unity3D>Unity3D在android下调试
查看>>