Estoy tratando de implementar el ajuste de círculo de mínimos cuadrados después del papel this (lo siento, no puedo publicarlo). El documento establece que podríamos caber un círculo, calculando el error geométrico como la distancia euclidiana (Xi '') entre un punto específico (Xi) y el punto correspondiente del círculo (Xi '). Tenemos tres parámetros: Xc (un vector de coordenadas del centro del círculo) y R (radio).Ajuste de círculo de mínimos cuadrados utilizando MATLAB Optimization Toolbox
me ocurrió con el siguiente código de MATLAB (en cuenta que estoy tratando de adaptarse a los círculos, y no las esferas como se indica en las imágenes):
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
El accesorio sin embargo no es demasiado bueno: si comienzo con el buen vector de parámetros, el algoritmo termina en el primer paso (por lo que hay un mínimo local donde debería estar), pero si perturbo el punto de inicio (con un círculo silencioso) la conexión se detiene con errores muy grandes Estoy seguro de que he pasado por alto algo en mi implementación.