opencv 直线拟合

一、拟合原理

opencv自带有最小二乘法拟合直线,但遇到有干扰点的时候,由于最小二乘法是将所有点代入矩阵进行求解,最终误差平方和最小即是最优直线,此时干扰点就会带入误差,效果很差。

随机采样一致算法,它是从一堆含干扰点的数据中,按照筛选条件划分为内点和外点,内点是组成模型的数据,外点是不适合模型的数据。
基本思想和流程如下:

bash 复制代码
1.给定一个数据集***S***,从中选择建立模型所需的最小样本数(空间直线最少可以由两个点确定,所以最小样本数是2,空间平面可以根据不共线三点确定,所以最小样本数为3,拟一个圆时,最小样本数是3),记选择数据集为***S1***
2.使用选择的数据集***S1***计算得到一个数学模型***M1***
3.用计算的模型***M1***去测试数据集中剩余的点,如果测试的数据点在误差允许的范围内,则将该数据点判为内点(inlier),否则判为外点(outlier),记所有内点组成的数据集为***S1****,S1* 称作 ***S1***的一致性集合
4.比较当前模型和之前推出的最好的模型的"内点"的数量,记录最大"内点"数量时模型参数和"内点"数量
5.重复1-4步,直到迭代结束或者当前模型已经足够好了("内点数目大于设定的阈值");每次产生的模型要么因为内点太少而被舍弃,要么因为比现有的模型更好而被选用

以RANSAC直线拟合为例,如下图所示,首先在点集中随机选择两个点,求解它们构成的直线参数,再计算点集中剩余点到该直线的距离,距离小于设置的距离阈值的点为内点,统计内点个数;接下来再随机选取两个点,同样统计内点个数,以此类推;其中内点个数最多的点集即为最大一致集.

此时意味着拟合的直线必然是随机两个点组成的直线,为了拟合精度更高,可以将最大一致集的内殿再使用最小二乘拟合出最终直线。

二、代码实现逻辑

cpp 复制代码
/// <summary>
/// ransac拟合直线
/// </summary>
/// <param name="points">输入点集</param>
/// <param name="line">输出直线方程</param>
/// <param name="iterations">拟合次数</param>
/// <param name="sigma">内点距离阈值</param>
/// <returns></returns>
bool fitLineRansac(const std::vector<cv::Point2f> points, cv::Vec4f& line, int iterations, double sigma)

三、效果

红色为最小二乘结果,绿色为ransac+最小二乘结果

当然,工程上还有一些简单粗暴的方法,按照所有点到拟合直线的距离进行筛选,筛选后再进行拟合。

获取源码请私信

相关推荐
我没胡说八道5 小时前
高校论文AI检测优化工具对比研究与实测分析(2026)
人工智能·深度学习·机器学习·计算机视觉·aigc·论文
探物 AI6 小时前
把 MambaOut 塞进 YOLOv11:会有什么样的反应
python·yolo·计算机视觉
_李小白11 小时前
【android opencv学习笔记】Day 32:直线检测之霍夫变换
android·opencv·学习
我最爱吃鱼香茄子13 小时前
终极方案:JetBrains IDE永久解放C盘空间
计算机视觉·性能优化·电脑·笔记本电脑·intellij-idea·程序员创富·webstorm
玖釉-14 小时前
Vulkan 离屏渲染详解:从 Framebuffer 到后处理、阴影贴图与 Render Texture
c++·windows·计算机视觉·图形渲染
路人甲32615 小时前
SONIC: Supersizing Motion Tracking for Natural Humanoid Whole-Body Control
人工智能·深度学习·计算机视觉·机器人·具身智能
程序员正茂15 小时前
EasyAR使用OpenCV下USB摄像头作为自定义相机
opencv·unity·easyar
_李小白15 小时前
【android opencv学习笔记】Day 31:提取轮廓之Canny算法
android·opencv·学习
armwind16 小时前
openISP学习8-GC — Gamma Correction(Gamma 校正)
图像处理·计算机视觉
大江东去浪淘尽千古风流人物16 小时前
【VGGT-Ω】前馈式3D重建的规模化之路:Register Attention、自监督训练与10B参数Scaling Law深度解析
深度学习·计算机视觉·transformer·slam·vio·3d重建