PCL专栏目录及须知
本算法一般用在寻找最优模型上,最大迭代次数往往要设置较大。
拟合方法 | 用时 | 样本点数 | 拟合得出内点数 | 拟合百分比 | 平面方程参数 |
---|---|---|---|---|---|
LMedS | 6.59863s | 34262 | 15880 | 46.3487% | A:-0.245971,B:0.328976,C:-0.911742,D:273.254 |
RMSAC | 40.2117s | 34262 | 2418 | 7.05738% | A:-0.14948,B:-0.570804,C:0.807365,D:-191.15 |
MSAC | 0.732149s | 34262 | 14551 | 42.4698% | A:-0.198437,B:0.323444,C:-0.925206,D:275.461 |
PROSAC | 0.0931069s | 34262 | 13627 | 39.7729% | A:0.256624,B:-0.288087,C:0.922578,D:-283.215 |
Ransac | 0.093296s | 34262 | 14870 | 43.4009% | A:-0.226923,B:0.322948,C:-0.918809,D:276.486 |
MLESAC | 4.8623s | 34262 | 13580 | 39.487% | A:-0.245971,B:0.328976,C:-0.911742,D:273.254 |
本文以拟合平面为例子,如果使用其他模型:修改该处代码,按照下文其他例程使用即可。
其他模型拟合例程:
1.Prosac 算法
(1)Prosac 算法是一种随机采样的RANSAC算法,用于解决拟合模型问题。
(2)核心思想:不断随机采样数据集,并在每次采样后计算型拟合误差,并根据误差来选择最优模型。
(3)优点:可以更快地找到最优模型,因为它能够在计算模型拟合误差的同时,随机采样数据集;
缺点:需要进行大量的计算和迭代,因此可能会有一定的运行时间。
(4)计算流程:
1)随机采样数据集。
2)计算模型拟合误差。
3)根据误差选择最优模型。
4)重复以上步骤直到达到预设的迭代次数或误差阀值。
2.完整代码
#include <pcl/io/pcd_io.h>
#include <pcl/point_types.h>
#include <pcl/point_cloud.h>
#include <pcl/sample_consensus/prosac.h>
#include <pcl/sample_consensus/sac_model_plane.h>
#include <pcl/visualization/pcl_visualizer.h>
#include <boost/thread/thread.hpp>
#include <pcl/console/time.h>
int main()
{
/****************PROSAC拟合(以拟合平面为例)********************/
pcl::PointCloud<pcl::PointXYZ>::Ptr cloud(new pcl::PointCloud<pcl::PointXYZ>); // 源点云
pcl::io::loadPCDFile("D:/code/csdn/data/samp11-TIN.pcd", *cloud);
pcl::SampleConsensusModelPlane<pcl::PointXYZ>::Ptr modelPlane(new pcl::SampleConsensusModelPlane<pcl::PointXYZ>(cloud)); // 待拟合平面点云对象
pcl::PointCloud<pcl::PointXYZ>::Ptr cloudInliers(new pcl::PointCloud<pcl::PointXYZ>); // 拟合出平面的内点
// PROSAC
pcl::console::TicToc time; time.tic();
pcl::ProgressiveSampleConsensus<pcl::PointXYZ> PROSAC(modelPlane); // PROSAC算法模型
PROSAC.setDistanceThreshold(3.0); // 距离阈值
PROSAC.setMaxIterations(500); // 最大迭代次数
PROSAC.computeModel(); // 拟合平面
Eigen::VectorXf PROSACCof;
PROSAC.getModelCoefficients(PROSACCof); // 获取模型参数
std::vector<int> PROSACInliers; // 获取属于拟合出平面的内点
PROSAC.getInliers(PROSACInliers);
pcl::copyPointCloud(*cloud, PROSACInliers, *cloudInliers);
std::cout << "PROSAC计算时间:" << time.toc() / 1000 << "s" << std::endl;
std::cout << "PROSAC原始点数:" << cloud->size() << std::endl;
std::cout << "PROSAC拟合点数:" << PROSACInliers.size() << std::endl;
std::cout << "PROSAC拟合百分比:" << ((double)PROSACInliers.size() / (double)cloud->size()) * 100 << "%" << std::endl;
std::cout << "PROSAC平面参数:" << "A:" << PROSACCof[0] << ",B:" << PROSACCof[1] << ",C:" << PROSACCof[2] << ",D:" << PROSACCof[3] << std::endl;
/****************展示********************/
boost::shared_ptr<pcl::visualization::PCLVisualizer> viewer(new pcl::visualization::PCLVisualizer("PROSAC"));
viewer->addPointCloud<pcl::PointXYZ>(cloud, "cloud");
viewer->addPointCloud<pcl::PointXYZ>(cloudInliers, pcl::visualization::PointCloudColorHandlerCustom<pcl::PointXYZ>(cloudInliers, 255, 0, 0), "cloudInliers");
while (!viewer->wasStopped())
{
viewer->spinOnce(100);
boost::this_thread::sleep(boost::posix_time::microseconds(100000));
}
return (0);
}
3.结果展示
打印:
可视化结果: