【发布时间】:2023-03-06 03:38:01
【问题描述】:
我正在尝试在this 论文之后实现最小二乘圆拟合(对不起,我无法发布它)。该论文指出,我们可以通过将几何误差计算为特定点 (Xi) 与圆上相应点 (Xi') 之间的欧式距离 (Xi'') 来拟合圆。我们有三个参数:Xc(圆心坐标向量)和 R(半径)。
我想出了以下 MATLAB 代码(请注意,我试图拟合圆形,而不是图像上所示的球体):
function [ circle ] = fit_circle( X )
% Kör paraméterstruktúra inicializálása
% R - kör sugara
% Xc - kör középpontja
circle.R = NaN;
circle.Xc = [ NaN; NaN ];
% Kezdeti illesztés
% A köz középpontja legyen a súlypont
% A sugara legyen az átlagos négyzetes távolság a középponttól
circle.Xc = mean( X );
d = bsxfun(@minus, X, circle.Xc);
circle.R = mean(bsxfun(@hypot, d(:,1), d(:,2)));
circle.Xc = circle.Xc(1:2)+random('norm', 0, 1, size(circle.Xc));
% Optimalizáció
options = optimset('Jacobian', 'on');
out = lsqnonlin(@ort_error, [circle.Xc(1), circle.Xc(2), circle.R], [], [], options, X);
end
%% Cost function
function [ error, J ] = ort_error( P, X )
%% Calculate error
R = P(3);
a = P(1);
b = P(2);
d = bsxfun(@minus, X, P(1:2)); % X - Xc
n = bsxfun(@hypot, d(:,1), d(:,2)); % || X - Xc ||
res = d - R * bsxfun(@times,d,1./n);
error = zeros(2*size(X,1), 1);
error(1:2:2*size(X,1)) = res(:,1);
error(2:2:2*size(X,1)) = res(:,2);
%% Jacobian
xdR = d(:,1)./n;
ydR = d(:,2)./n;
xdx = bsxfun(@plus,-R./n+(d(:,1).^2*R)./n.^3,1);
ydy = bsxfun(@plus,-R./n+(d(:,2).^2*R)./n.^3,1);
xdy = (d(:,1).*d(:,2)*R)./n.^3;
ydx = xdy;
J = zeros(2*size(X,1), 3);
J(1:2:2*size(X,1),:) = [ xdR, xdx, xdy ];
J(2:2:2*size(X,1),:) = [ ydR, ydx, ydy ];
end
但是拟合不太好:如果我从好的参数向量开始,算法会在第一步终止(所以应该有一个局部最小值),但是如果我扰乱了起点(无噪音圈)拟合以非常大的错误停止。我确信我在实现中忽略了一些东西。
【问题讨论】:
标签: matlab mathematical-optimization least-squares bsxfun