Python十行代码实现简单卡尔曼滤波(Kalman Filter)

关键代码

import numpy as np
#一步预测
def kf_predict(X0, P0, A, Q, B, U1):
    X10 = np.dot(A,X0) + np.dot(B,U1)
    P10 = np.dot(np.dot(A,P0),A.T)+ Q
    return (X10, P10)
#测量更新
def kf_update(X10, P10, Z, H, R):
    K = np.dot(np.dot(P10,H.T),np.linalg.pinv(np.dot(np.dot(H,P10),H.T) + R))
    X1 = X10 + np.dot(K,Z - np.dot(H,X10))
    P1 = np.dot(np.eye(K.shape[0]) - np.dot(K,H),P10)
    return (X1, P1, K)

解释

离散的状态方程、观测方程及它们的随机过程如下:
X ( k ) = A X ( k − 1 ) + B U ( k ) + w ( k − 1 ) Z ( k ) = H X ( k ) + e ( k ) p ( w ) = N ( 0 , Q ) p ( e ) = N ( 0 , R ) X(k) = AX(k-1) + BU(k) + w(k-1)\\ Z(k) = HX(k) + e(k)\\ p(w) = N(0, Q)\\ p(e) = N(0, R)\\ X(k)=AX(k1)+BU(k)+w(k1)Z(k)=HX(k)+e(k)p(w)=N(0,Q)p(e)=N(0,R)

如果是连续的状态方程则需要离散化,例如上式中的A等于: A = e x p m ( F Δ t ) A = expm(F\Delta t) A=expm(FΔt)
其中expm指矩阵指数,F为微分运动方程 X ˙ = F X \dot X=FX X˙=FX线性化后的系数矩阵,可以使用sympy.exp协助推导。(线性化及离散化不属于本文范围)

Kalman Filter主要步骤为一步预测和测量更新两个部分,以下列出Kalman黄金5公式
一步预测
X ( k , k − 1 ) = A X ( k − 1 ) + B U ( k ) P ( k , k − 1 ) = A P ( k − 1 ) A T + Q X(k,k-1)=AX(k-1)+BU(k)\\ P(k,k-1)=AP(k-1)A^T+Q X(k,k1)=AX(k1)+BU(k)P(k,k1)=AP(k1)AT+Q
测量更新:
K ( k ) = P ( k , k − 1 ) H T [ H P ( k , k − 1 ) H T + R ] − 1 X ( k ) = X ( k , k − 1 ) + K ( k ) [ Z ( k ) − H X ( k , k − 1 ) ] P ( k ) = [ I − K ( k ) H ] P ( k , k − 1 ) K(k)=P(k,k-1)H^T[HP(k,k-1)H^T+R]^{-1}\\ X(k)=X(k,k-1)+K(k)[Z(k)-HX(k,k-1)]\\ P(k)=[I-K(k)H]P(k,k-1) K(k)=P(k,k1)HT[HP(k,k1)HT+R]1X(k)=X(k,k1)+K(k)[Z(k)HX(k,k1)]P(k)=[IK(k)H]P(k,k1)

import numpy as np
#一步预测
'''
设状态量有xn个
- X0为前一时刻状态量,shape=(xn,1)
- P0为初始状态不确定度, shape=(xn,xn)
- A为状态转移矩阵,shape=(xn,xn)
- Q为递推噪声协方差矩阵,shape=(xn,xn)
- B、U1是外部输入部分

返回的结果为
- X10为一步预测的状态量结果,shape=(xn,1)
- P10为一步预测的协方差,shape=(xn,xn)
'''
def kf_predict(X0, P0, A, Q, B, U1):
    X10 = np.dot(A,X0) + np.dot(B,U1)
    P10 = np.dot(np.dot(A,P0),A.T)+ Q
    return (X10, P10)

'''
设状态量有xn个
- X10为一步预测的状态量结果,shape=(xn,1)
- P10为一步预测的协方差,shape=(xn,xn)
- Z为观测值,shape=(xn,1)
- H为观测系数矩阵,shape=(xn,xn)
- R为观测噪声协方差,shape=(xn,xn)

返回的结果为
- X1为一步预测的状态量结果,shape=(xn,1)
- P1为一步预测的协方差,shape=(xn,xn)
- K为卡尔曼增益,不需要返回,但是可以看一下它的值来判断是否正常运行
'''
#测量更新
def kf_update(X10, P10, Z, H, R):
    K = np.dot(np.dot(P10,H.T),np.linalg.pinv(np.dot(np.dot(H,P10),H.T) + R))
    X1 = X10 + np.dot(K,Z - np.dot(H,X10))
    P1 = np.dot(np.eye(K.shape[0]) - np.dot(K,H),P10)
    return (X1, P1, K)

注意在Numpy shape(n,) 不等于shape(n,1)

例子

以匀加速度运动为例,结果如下,代码在最后

在这里插入图片描述

可见前期偏预测、后期偏观测

# -*- coding: utf-8 -*-
"""
Created on Mon Jul  4 17:15:01 2022

@author: ThinkPad
"""



import numpy as np
import matplotlib.pyplot as plt

"""
X(k) = AX(k-1) + BU(k) + w(k-1)
Z(k) = HX(k) + e(k)
p(w) = N(0, Q)
p(e) = N(0, R)
"""

def kf_predict(X0, P0, A, Q, B, U1):
    X10 = np.dot(A,X0) + np.dot(B,U1)
    P10 = np.dot(np.dot(A,P0),A.T)+ Q
    return (X10, P10)


        
def kf_update(X10, P10, Z, H, R):
    V = Z - np.dot(H,X10)
    K = np.dot(np.dot(P10,H.T),np.linalg.pinv(np.dot(np.dot(H,P10),H.T) + R))
    X1 = X10 + np.dot(K,V)
    P1 = np.dot(np.eye(K.shape[0]) - np.dot(K,H),P10)
    return (X1, P1, K)

"""
加速度白噪声建模
状态方程:
x' = v'
v' = a'
a' = 0 
离散化得到;
x(k) = x(k-1)+t*v(k)+0.5*t^2*a(k)
v(k) = v(k-1)+t*a(k)
a(k) = a(k-1)

观测方程:
z(k) = x(k) + e

"""

n = 20 #数据量
nx = 3 #变量数量
t = np.linspace(0,3,n) #时间序列
dt = t[1] - t[0]

#真实函数关系
a_true = np.ones(n)*9.8 + np.random.normal(0,1,size=n)
v_true = np.cumsum(a_true*dt)
x_true = np.cumsum(v_true*dt)
X_true = np.concatenate([x_true, v_true, a_true]).reshape([nx,-1])


# 观测噪声协方差!!!!!!!!!!!!!!!!!!!!(可调整)
R = np.diag([10**2])

#仿真观测值
e = np.random.normal(0,2,n)
x_obs = x_true + e

# 计算系数
A = np.array([1,dt,0.5*dt**2,
              0,1,dt,
              0,0,1]).reshape([nx,nx])
B = 0
U1 = 0

#状态假设(观测)初始值
x0 = 0
v0 = 0
a0 = 10.0
X0 = np.array([x0,v0,a0]).reshape(nx,1)

#初始状态不确定度!!!!!!!!!!!!!!!!(可调整)
P0 = np.diag([0**2,0**2,0.2**2])

#状态递推噪声协方差!!!!!!!!!!!!!!!!!!(可调整)
Q = np.diag([0.0**2,0**2,1**2])

###开始处理
X1_np = np.copy(X0)
P1_list = [P0]
X10_np = np.copy(X0)
P10_list = [P0]

for i in range(n):
    Zi = np.array(x_obs[i]).reshape([1,1])
    Hi = np.array([1,0,0]).reshape([1,nx])
    
    if (i == 0):
        continue
    else:
        Xi = X1_np[:,i-1].reshape([nx,1])
        Pi = P1_list[i-1]
        X10, P10 = kf_predict(Xi, Pi, A, Q, B, U1)

        X10_np = np.concatenate([X10_np, X10], axis=1)
        P10_list.append(P10)

        X1, P1, K = kf_update(X10, P10, Zi, Hi, R)
        X1_np = np.concatenate([X1_np, X1], axis=1)
        P1_list.append(P1)

#结束,绘图
fig = plt.figure()
ax1 = fig.add_subplot(1,1,1)
ax1.plot(x_true, 'k-', label="Truth")
ax1.plot(X1_np[0,:], 'go--', label="Kalman Filter")
ax1.scatter(np.arange(n), x_obs, label="Observation", marker='*')

plt.legend()
plt.show()

  • 57
    点赞
  • 328
    收藏
    觉得还不错? 一键收藏
  • 14
    评论
以下是Python实现对WAV信号的卡尔曼滤波的示例代码: ```python import numpy as np import scipy.io.wavfile as wav import matplotlib.pyplot as plt # 读取WAV文件 fs, x = wav.read('signal.wav') # 卡尔曼滤波器 def kalman_filter(z_meas, x_esti, P): # 状态转移矩阵 F = np.array([[1, 1], [0, 1]]) # 系统噪声方差 Q = np.array([[0.05, 0.05], [0.05, 0.2]]) # 观测矩阵 H = np.array([1, 0]).reshape(1, 2) # 观测噪声方差 R = np.array([0.2]).reshape(1, 1) # 状态预测 x_predict = np.dot(F, x_esti) P_predict = np.dot(F, np.dot(P, F.T)) + Q # 计算卡尔曼增益 K = np.dot(P_predict, np.dot(H.T, np.linalg.inv(np.dot(H, np.dot(P_predict, H.T)) + R))) # 更新状态估计值和误差协方差矩阵 x_estimate = x_predict + np.dot(K, (z_meas - np.dot(H, x_predict))) P_estimate = np.dot((np.eye(2) - np.dot(K, H)), P_predict) return x_estimate, P_estimate # 初始化 x_esti = np.array([0, 0]).reshape(2, 1) P = 1.0 * np.eye(2) y_filt = np.zeros_like(x) # 卡尔曼滤波 for i in range(len(x)): z_meas = np.array([x[i], 0]).reshape(2, 1) x_esti, P = kalman_filter(z_meas, x_esti, P) y_filt[i] = x_esti[0] # 绘制原始信号和滤波后的信号 plt.subplot(2, 1, 1) plt.plot(x) plt.title('Original Signal') plt.subplot(2, 1, 2) plt.plot(y_filt) plt.title('Kalman Filtered Signal') plt.tight_layout() plt.show() ``` 这段代码首先使用scipy库中的wavfile模块读取WAV文件,然后定义了一个卡尔曼滤波器函数kalman_filter,用来处理每个采样点的数据。在主循环中,对于每个采样点,卡尔曼滤波器函数kalman_filter都会被调用一次,得到滤波后的输出值,并将其保存到y_filt数组中。最后,使用matplotlib库绘制原始信号和滤波后的信号的图形。 需要注意的是,卡尔曼滤波器的性能取决于系统噪声、观测噪声和初始状态估计的准确性。在实际应用中,需要根据具体情况进行调整和优化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值