RANSAC算法

算法基本思想和流程

RANSAC是通过反复选择数据集去估计出模型,一直迭代到估计出认为比较好的模型。
具体的实现步骤可以分为以下几步:

  1. 选择出可以估计出模型的最小数据集;(对于直线拟合来说就是两个点,对于计算Homography矩阵就是4个点)
  2. 使用这个数据集来计算出数据模型;
  3. 将所有数据带入这个模型,计算出“内点”的数目;(累加在一定误差范围内的适合当前迭代推出模型的数据)
  4. 比较当前模型和之前推出的最好的模型的“内点“的数量,记录最大“内点”数的模型参数和“内点”数;
  5. 重复1-4步,直到迭代结束或者当前模型已经足够好了(“内点数目大于一定数量”)。

迭代次数推导

这里有一点就是迭代的次数我们应该选择多大呢?这个值是否可以事先知道应该设为多少呢?还是只能凭经验决定呢? 这个值其实是可以估算出来的。下面我们就来推算一下。

假设“内点”在数据中的占比为 [公式]

[公式]

那么我们每次计算模型使用 [公式] 个点的情况下,选取的点至少有一个外点的情况就是

[公式]

也就是说,在迭代 [公式] 次的情况下, [公式] 就是 [公式] 次迭代计算模型都至少采样到一个“外点”去计算模型的概率。那么能采样到正确的 [公式] 个点去计算出正确模型的概率就是

[公式]

通过上式,可以求得

[公式]

“内点”的概率  [公式] 通常是一个先验值。然后  [公式] 是我们希望RANSAC得到正确模型的概率。如果事先不知道  [公式] 的值,可以使用自适应迭代次数的方法。也就是一开始设定一个无穷大的迭代次数,然后每次更新模型参数估计的时候,用当前的“内点”比值当成  [公式] 来估算出迭代次数。

用Python实现直线拟合

import numpy as np
import matplotlib.pyplot as plt
import random
import math

# 数据量。
SIZE = 50
# 产生数据。np.linspace 返回一个一维数组,SIZE指定数组长度。
# 数组最小值是0,最大值是10。所有元素间隔相等。
X = np.linspace(0, 10, SIZE)
Y = 3 * X + 10

fig = plt.figure()
# 画图区域分成1行1列。选择第一块区域。
ax1 = fig.add_subplot(1,1, 1)
# 标题
ax1.set_title("RANSAC")


# 让散点图的数据更加随机并且添加一些噪声。
random_x = []
random_y = []
# 添加直线随机噪声
for i in range(SIZE):
    random_x.append(X[i] + random.uniform(-0.5, 0.5)) 
    random_y.append(Y[i] + random.uniform(-0.5, 0.5)) 
# 添加随机噪声
for i in range(SIZE):
    random_x.append(random.uniform(0,10))
    random_y.append(random.uniform(10,40))
RANDOM_X = np.array(random_x) # 散点图的横轴。
RANDOM_Y = np.array(random_y) # 散点图的纵轴。

# 画散点图。
ax1.scatter(RANDOM_X, RANDOM_Y)
# 横轴名称。
ax1.set_xlabel("x")
# 纵轴名称。
ax1.set_ylabel("y")

# 使用RANSAC算法估算模型
# 迭代最大次数,每次得到更好的估计会优化iters的数值
iters = 100000
# 数据和模型之间可接受的差值
sigma = 0.25
# 最好模型的参数估计和内点数目
best_a = 0
best_b = 0
pretotal = 0
# 希望的得到正确模型的概率
P = 0.99
for i in range(iters):
    # 随机在数据中红选出两个点去求解模型
    sample_index = random.sample(range(SIZE * 2),2)
    x_1 = RANDOM_X[sample_index[0]]
    x_2 = RANDOM_X[sample_index[1]]
    y_1 = RANDOM_Y[sample_index[0]]
    y_2 = RANDOM_Y[sample_index[1]]

    # y = ax + b 求解出a,b
    a = (y_2 - y_1) / (x_2 - x_1)
    b = y_1 - a * x_1

    # 算出内点数目
    total_inlier = 0
    for index in range(SIZE * 2):
        y_estimate = a * RANDOM_X[index] + b
        if abs(y_estimate - RANDOM_Y[index]) < sigma:
            total_inlier = total_inlier + 1

    # 判断当前的模型是否比之前估算的模型好
    if total_inlier > pretotal:
        iters = math.log(1 - P) / math.log(1 - pow(total_inlier / (SIZE * 2), 2))
        pretotal = total_inlier
        best_a = a
        best_b = b

    # 判断是否当前模型已经符合超过一半的点
    if total_inlier > SIZE:
        break

# 用我们得到的最佳估计画图
Y = best_a * RANDOM_X + best_b

# 直线图
ax1.plot(RANDOM_X, Y)
text = "best_a = " + str(best_a) + "\nbest_b = " + str(best_b)
plt.text(5,10, text,
         fontdict={'size': 8, 'color': 'r'})
plt.show()

 

### RANSAC算法概述 RANSAC(Random Sample Consensus,随机采样一致性)算法是一种基于随机采样的迭代算法,主要用于从一组包含大量噪声和异常值(外点)的数据中估计数学模型的参数。该算法由Fischler和Bolles于1981年提出,并在计算机视觉和计算机图形学等领域得到了广泛应用[^2]。 ### RANSAC算法原理 RANSAC的核心思想是从数据集中随机选取最小数量的样本集来构建一个可能的模型假设;然后计算其他数据点与这个假设模型的一致性程度,即内点的数量;重复上述过程多次,最终选择具有最多一致性的模型作为最佳模型。具体来说: - **初始化**:设定最大迭代次数`N`、阈值`t`用于判断内外点的标准。 - **迭代过程**: - 随机抽取最少必要量的数据子集形成初始模型; - 计算剩余所有数据相对于此模型的距离误差; - 将距离小于给定阈值`t`的数据视为内点集合; - 更新当前最好的模型及其对应的内点数。 - **终止条件**:当达到预设的最大迭代次数或找到足够好的模型时结束循环。 ### Python实现示例 下面给出一段简单的Python代码实现了针对二维直线拟合场景下的RANSAC算法: ```python import numpy as np from sklearn.linear_model import LinearRegression def ransac_fit_line(points, n_iterations=100, threshold=1e-2): best_inliers = None max_inliers_count = 0 for _ in range(n_iterations): # Randomly select two points to form a line model sample_indices = np.random.choice(len(points), size=2, replace=False) sampled_points = points[sample_indices] # Fit the linear regression on these samples lr = LinearRegression().fit(sampled_points[:, :-1], sampled_points[:, -1]) # Predict distances of all other points from this fitted line predictions = lr.predict(points[:, :-1]) residuals = abs(predictions - points[:, -1]) # Count how many are within our tolerance level (inliers) current_inliers = points[residuals < threshold] num_current_inliers = len(current_inliers) if num_current_inliers > max_inliers_count: best_inliers = current_inliers.copy() max_inliers_count = num_current_inliers return best_inliers # Example usage with synthetic data containing outliers np.random.seed(42) X = np.linspace(-5, 5, 100).reshape((-1, 1)) y_true = X * 3 + 7 noise = np.random.normal(size=y_true.shape)*2 outlier_mask = np.zeros_like(y_true,dtype=bool) outlier_mask[:10]=True y_noisy = y_true + noise y_outliers = y_noisy.copy() y_outliers[outlier_mask]+=10*(np.random.rand(sum(outlier_mask))-0.5) data_with_outliers=np.hstack([X,y_outliers]) best_fitted_data=ransac_fit_line(data_with_outliers,n_iterations=100,threshold=.5) print('Best Fitted Data Points:\n',best_fitted_data) ``` 这段代码展示了如何利用Scikit-Learn库中的LinearRegression类来进行简单线性回归建模,并通过自定义函数`ransac_fit_line()`实现了基本形式的RANSAC算法逻辑[^1]。 ### 应用领域 除了经典的线性和平面拟合之外,RANSAC还广泛应用于以下几个方面: - 图像拼接中去除误匹配的关键点对,提高图像间变换矩阵估算精度[^4]。 - 在三维重建任务里用来识别并剔除激光扫描仪获取到的离群点云数据。 - 自动驾驶车辆感知模块中处理传感器输入信号中存在的干扰因素。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值