matlab练习程序(无迹变换)

当数据通过非线性函数后,分布不再是高斯分布时,可以用无迹变换估计新数据的均值与方差。

算法原理就是在原始数据均值周围根据方差选取一些待使用点,然后将这些点通过非线性函数,再通过加权平均的方式求出新分布的均值与方差。

如果我们选取的点非常多,并且将这些点都通过非线性函数,再估计均值与方差,那么就是粒子滤波了。

无迹变换的典型算法就是无迹卡尔曼滤波了,我这里只是单纯实验了无迹变换,没有涉及卡尔曼滤波部分,后面有时间应该会写无迹卡尔曼滤波。

无迹变换中选取加权的权重公式我看还挺复杂,并且我不是完全清楚其意义,所以这里直接暴力取权重了,对于实验数据效果还可以。 

matlab代码如下:

clear all;close all;clc;

% 非线性函数:
% xpar = x+y
% ypar = 5*x*x +y*y

x = randn(1000,1)*2;
y = randn(1000,1);

mu = [mean(x);mean(y)];
P = [x y]'*[x y] / length(x);

plot(x,y,'r.')
hold on;
plot(mu(1),mu(2),'b*')

%%%%%%%%%%%%ut变换%%%%%%%%%%%%%%%%%%%%
n = 2;
N = 2*n+1;
Wm = ones(1, N) * 1/(2*(n+1));                      %这里比较暴力直接取权重
Wc = [Wm(1), Wm(2:end)+ones(1, 2*n)*1/(2*(n+1))];

%{
alpha = 0.9;                  %根据公式算权重,需要仔细考虑各个系数
kappa = 2;
beta = 2;
lambda = alpha^2 * (n + kappa) - n;
Wm = zeros(1,2*n+1);
Wc = zeros(1,2*n+1);
Wm(1) = lambda / (n + lambda);
Wc(1) = Wm(1) + (1 - alpha^2 + beta);
for i = 2 : 2*n+1
    Wm(i) = 1 / (2 * (n + lambda));
    Wc(i) = Wm(i);
end
%}

L = chol(P, 'lower');
X = [mu, mu + sqrt(n+1)*L, mu - sqrt(n+1)*L];       %得到原始均值周围的点作为待处理点

Y = zeros(size(X));                                 %待处理点过非线性函数
for i = 1:N
    newx = X(1,i);
    newy = X(2,i);
    Y(1,i) = newx+newy;
    Y(2,i) = 5*newx.^2+newy.^2;
end

mu_hat = Y * Wm';                                   %均值加权平均
P_hat = (Y - mu_hat) * diag(Wc) * (Y - mu_hat)';    %方差加权平均

for i=1:N
  plot(X(1,i),X(2,i),'bo');
end
legend('原始数据点','原始数据均值','ut变换选取的待处理点');
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%

xpar = x + y;                     %如果所有数据点都过非线性函数再求均值,那就是粒子滤波了
ypar = 5*x.^2+y.^2;

figure;
plot(xpar,ypar,'r.')
hold on;
plot(mean(xpar),mean(ypar),'g*')

xmean = mu(1) + mu(2);            %原始数据点的均值过非线性函数得到的新分布的均值是不对的
ymean = 5*mu(1).^2 + mu(2)^2;
plot(xmean,ymean,'b*');

for i=1:N
  plot(Y(1,i),Y(2,i),'bo');
end
plot(mu_hat(1),mu_hat(2),'go');

legend('非线性函数处理后的原始数据点','非线性函数处理后的原始数据点的均值', ...
'原始数据点的均值过非线性函数后的值','ut变换选取的待处理点过非线性函数后的值','ut变换最终结果(应该是绿圈,算bug吗。。。)');

结果如下:

posted @ 2023-06-22 14:49  Dsp Tian  阅读(341)  评论(0编辑  收藏  举报