LowLevelParticleFilters
This is a library for state estimation, that is, given measurements $y(t)$ from a dynamical system, estimate the state vector $x(t)$. Throughout, we assume dynamics on the form
\[\begin{aligned} x(t+1) &= f(x(t), u(t), p, t, w(t))\\ y(t) &= g(x(t), u(t), p, t, e(t)) \end{aligned}\]
or the linear version
\[\begin{aligned} x(t+1) &= Ax(t) + Bu(t) + w(t)\\ y(t) &= Cx(t) + Du(t) + e(t) \end{aligned}\]
where $x$ is the state vector, $u$ an input, $p$ some form of parameters, $t$ is the time and $w,e$ are disturbances (noise). Throughout the documentation, we often call the function $f$ dynamics
and the function $g$ measurement
.
The dynamics above describe a discrete-time system, i.e., the function $f$ takes the current state and produces the next state. This is in contrast to a continuous-time system, where $f$ takes the current state but produces the time derivative of the state. A continuous-time system can be discretized, described in detail in Discretization.
The parameters $p$ can be anything, or left out. You may write the dynamics functions such that they depend on $p$ and include parameters when you create a filter object. You may also override the parameters stored in the filter object when you call any function on the filter object. This behavior is modeled after the SciML ecosystem.
Depending on the nature of $f$ and $g$, the best method of estimating the state may vary. If $f,g$ are linear and the disturbances are additive and Gaussian, the KalmanFilter
is an optimal state estimator. If any of the above assumptions fail to hold, we may need to resort to more advanced estimators. This package provides several filter types, outlined below.
Estimator types
We provide a number of filter types
KalmanFilter
. A standard Kalman filter. Is restricted to linear dynamics (possibly time varying) and Gaussian noise.SqKalmanFilter
. A standard Kalman filter on square-root form (slightly slower but more numerically stable with ill-conditioned covariance).ExtendedKalmanFilter
: For nonlinear systems, the EKF runs a regular Kalman filter on linearized dynamics. Uses ForwardDiff.jl for linearization (or user provided). The noise model must still be Gaussian and additive.UnscentedKalmanFilter
: The Unscented Kalman filter often performs slightly better than the Extended Kalman filter but may be slightly more computationally expensive. The UKF handles nonlinear dynamics and measurement models, but still requires a Gaussian noise model (may be non additive) and still assumes that all posterior distributions are Gaussian, i.e., can not handle multi-modal posteriors.ParticleFilter
: The particle filter is a nonlinear estimator. This version of the particle filter is simple to use and assumes that both dynamics noise and measurement noise are additive. Particle filters handle multi-modal posteriors.AdvancedParticleFilter
: This filter gives you more flexibility, at the expense of having to define a few more functions. This filter does not require the noise to be additive and is thus the most flexible filter type.AuxiliaryParticleFilter
: This filter is identical toParticleFilter
, but uses a slightly different proposal mechanism for new particles.IMM
: (Currently considered experimental) The Interacting Multiple Models filter switches between multiple internal filters based on a hidden Markov model. This filter is useful when the system dynamics change over time and the change can be modeled as a discrete Markov chain, i.e., the system may switch between a small number of discrete "modes".
Functionality
This package provides
- Filtering, estimating $x(t)$ given measurements up to and including time $t$. We call the filtered estimate $x(t|t)$ (read as $x$ at $t$ given $t$).
- Smoothing, estimating $x(t)$ given data up to $T > t$, i.e., $x(t|T)$.
- Parameter estimation.
All filters work in two distinct steps.
- The prediction step (
predict!
). During prediction, we use the dynamics model to form $x(t|t-1) = f(x(t-1), ...)$ - The correction step (
correct!
). In this step, we adjust the predicted state $x(t|t-1)$ using the measurement $y(t)$ to form $x(t|t)$.
(The IMM filter is an exception to the above and has two additional steps, combine!
and interact!
)
In general, all filters represent not only a point estimate of $x(t)$, but a representation of the complete posterior probability distribution over $x$ given all the data available up to time $t$. One major difference between different filter types is how they represent these probability distributions.
Particle filter
A particle filter represents the probability distribution over the state as a collection of samples, each sample is propagated through the dynamics function $f$ individually. When a measurement becomes available, the samples, called particles, are given a weight based on how likely the particle is given the measurement. Each particle can thus be seen as representing a hypothesis about the current state of the system. After a few time steps, most weights are inevitably going to be extremely small, a manifestation of the curse of dimensionality, and a resampling step is incorporated to refresh the particle distribution and focus the particles on areas of the state space with high posterior probability.
Defining a particle filter (ParticleFilter
) is straightforward, one must define the distribution of the noise df
in the dynamics function, dynamics(x,u,p,t)
and the noise distribution dg
in the measurement function measurement(x,u,p,t)
. Both of these noise sources are assumed to be additive, but can have any distribution (see AdvancedParticleFilter
for non-additive noise). The distribution of the initial state estimate d0
must also be provided. In the example below, we use linear Gaussian dynamics so that we can easily compare both particle and Kalman filters. (If we have something close to linear Gaussian dynamics in practice, we should of course use a Kalman filter and not a particle filter.)
using LowLevelParticleFilters, LinearAlgebra, StaticArrays, Distributions, Plots
Define problem
nx = 2 # Dimension of state
nu = 1 # Dimension of input
ny = 1 # Dimension of measurements
N = 500 # Number of particles
const dg = MvNormal(ny,0.2) # Measurement noise Distribution
const df = MvNormal(nx,0.1) # Dynamics noise Distribution
const d0 = MvNormal(randn(nx),2.0) # Initial state Distribution
Define linear state-space system (using StaticArrays for maximum performance)
const A = SA[0.97043 -0.097368
0.09736 0.970437]
const B = SA[0.1; 0;;]
const C = SA[0 1.0]
Next, we define the dynamics and measurement equations, they both take the signature (x,u,p,t) = (state, input, parameters, time)
dynamics(x,u,p,t) = A*x .+ B*u
measurement(x,u,p,t) = C*x
vecvec_to_mat(x) = copy(reduce(hcat, x)') # Helper function
the parameter p
can be anything, and is often optional. If p
is not provided when performing operations on filters, any p
stored in the filter objects (if supported) is used. The default if none is provided and none is stored in the filter is p = LowLevelParticleFilters.NullParameters()
.
We are now ready to define and use a filter
pf = ParticleFilter(N, dynamics, measurement, df, dg, d0)
ParticleFilter{PFstate{StaticArraysCore.SVector{2, Float64}, Float64}, typeof(Main.dynamics), typeof(Main.measurement), Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}, Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}, Distributions.IsoNormal, DataType, Random.Xoshiro, LowLevelParticleFilters.NullParameters}
state: PFstate{StaticArraysCore.SVector{2, Float64}, Float64}
dynamics: dynamics (function of type typeof(Main.dynamics))
measurement: measurement (function of type typeof(Main.measurement))
dynamics_density: Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}
measurement_density: Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}
initial_density: Distributions.IsoNormal
resample_threshold: Float64 0.1
resampling_strategy: LowLevelParticleFilters.ResampleSystematic <: LowLevelParticleFilters.ResamplingStrategy
rng: Random.Xoshiro
p: LowLevelParticleFilters.NullParameters LowLevelParticleFilters.NullParameters()
threads: Bool false
Ts: Float64 1.0
With the filter in hand, we can simulate from its dynamics and query some properties
du = MvNormal(nu,1.0) # Random input distribution for simulation
xs,u,y = simulate(pf,200,du) # We can simulate the model that the pf represents
pf(u[1], y[1]) # Perform one filtering step using input u and measurement y
particles(pf) # Query the filter for particles, try weights(pf) or expweights(pf) as well
x̂ = weighted_mean(pf) # using the current state
2-element Vector{Float64}:
-0.0755112853969465
-0.18985543111315506
If you want to perform batch filtering using an existing trajectory consisting of vectors of inputs and measurements, try any of the functions forward_trajectory
, mean_trajectory
:
sol = forward_trajectory(pf, u, y) # Filter whole trajectories at once
x̂,ll = mean_trajectory(pf, u, y)
plot(sol, xreal=xs, markersize=2)
u
ad y
are then assumed to be vectors of vectors. StaticArrays is recommended for maximum performance.
If MonteCarloMeasurements.jl is loaded, you may transform the output particles to Matrix{MonteCarloMeasurements.Particles}
with the layout T × n_state
using Particles(x,we)
. Internally, the particles are then resampled such that they all have unit weight. This is conventient for making use of the plotting facilities of MonteCarloMeasurements.jl.
For a full usage example, see the benchmark section below or example_lineargaussian.jl
Resampling
The particle filter will perform a resampling step whenever the distribution of the weights has become degenerate. The resampling is triggered when the effective number of samples is smaller than pf.resample_threshold
$\in [0, 1]$, this value can be set when constructing the filter. How the resampling is done is governed by pf.resampling_strategy
, we currently provide ResampleSystematic <: ResamplingStrategy
as the only implemented strategy. See https://en.wikipedia.org/wiki/Particle_filter for more info.
Particle Smoothing
Smoothing is the process of finding the best state estimate given both past and future data. Smoothing is thus only possible in an offline setting. This package provides a particle smoother, based on forward filtering, backward simulation (FFBS), example usage follows:
N = 2000 # Number of particles
T = 80 # Number of time steps
M = 100 # Number of smoothed backwards trajectories
pf = ParticleFilter(N, dynamics, measurement, df, dg, d0)
du = MvNormal(nu,1) # Control input distribution
x,u,y = simulate(pf,T,du) # Simulate trajectory using the model in the filter
tosvec(y) = reinterpret(SVector{length(y[1]),Float64}, reduce(hcat,y))[:] |> copy
x,u,y = tosvec.((x,u,y)) # It's good for performance to use StaticArrays to the extent possible
xb,ll = smooth(pf, M, u, y) # Sample smoothing particles
xbm = smoothed_mean(xb) # Calculate the mean of smoothing trajectories
xbc = smoothed_cov(xb) # And covariance
xbt = smoothed_trajs(xb) # Get smoothing trajectories
xbs = [diag(xbc) for xbc in xbc] |> vecvec_to_mat .|> sqrt
plot(xbm', ribbon=2xbs, lab="PF smooth")
plot!(vecvec_to_mat(x), l=:dash, lab="True")
We can plot the particles themselves as well
downsample = 5
plot(vecvec_to_mat(x), l=(4,), layout=(2,1), show=false)
scatter!(xbt[1, 1:downsample:end, :]', subplot=1, show=false, m=(1,:black, 0.5), lab="")
scatter!(xbt[2, 1:downsample:end, :]', subplot=2, m=(1,:black, 0.5), lab="")
Kalman filter
The KalmanFilter
(wiki) assumes that $f$ and $g$ are linear functions, i.e., that they can be written on the form
\[\begin{aligned} x(t+1) &= Ax(t) + Bu(t) + w(t)\\ y(t) &= Cx(t) + Du(t) + e(t) \end{aligned}\]
for some matrices $A,B,C,D$ where $w \sim N(0, R_1)$ and $e \sim N(0, R_2)$ are zero mean and Gaussian. The Kalman filter represents the posterior distributions over $x$ by the mean and a covariance matrix. The magic behind the Kalman filter is that linear transformations of Gaussian distributions remain Gaussian, and we thus have a very efficient way of representing them.
A Kalman filter is easily created using the constructor KalmanFilter
. Many of the functions defined for particle filters, are defined also for Kalman filters, e.g.:
R1 = cov(df)
R2 = cov(dg)
kf = KalmanFilter(A, B, C, 0, R1, R2, d0)
sol = forward_trajectory(kf, u, y) # sol contains filtered state, predictions, pred cov, filter cov, loglik
It can also be called in a loop like the pf
above
for t = 1:T
kf(u,y) # Performs both correct! and predict!
# alternatively
ll, e = correct!(kf, y, nothing, t) # Returns loglikelihood and prediction error (plus other things if you want)
x = state(kf) # Access the state estimate
R = covariance(kf) # Access the covariance of the estimate
predict!(kf, u, nothing, t)
end
The matrices in the Kalman filter may be time varying, such that A[:, :, t]
is $A(t)$. They may also be provided as functions on the form $A(t) = A(x, u, p, t)$. This works for both dynamics and covariance matrices.
The numeric type used in the Kalman filter is determined from the mean of the initial state distribution, so make sure that this has the correct type if you intend to use, e.g., Float32
or ForwardDiff.Dual
for automatic differentiation.
Smoothing using KF
Kalman filters can also be used for smoothing
kf = KalmanFilter(A, B, C, 0, cov(df), cov(dg), d0)
xT,R,lls = smooth(kf, u, y) # Returns smoothed state, smoothed cov, loglik
Plot and compare PF and KF
plot(vecvec_to_mat(xT), lab="Kalman smooth", layout=2)
plot!(xbm', lab="pf smooth")
plot!(vecvec_to_mat(x), lab="true")
Kalman filter tuning tutorial
The tutorial "How to tune a Kalman filter" details how to figure out appropriate covariance matrices for the Kalman filter, as well as how to add disturbance models to the system model.
Unscented Kalman Filter
The UnscentedKalmanFilter
represents posterior distributions over $x$ as Gaussian distributions just like the KalmanFilter
, but propagates them through a nonlinear function $f$ by a deterministic sampling of a small number of particles called sigma points (this is referred to as the unscented transform). This UKF thus handles nonlinear functions $f,g$, but only Gaussian disturbances and unimodal posteriors. The UKF will by default treat the noise as additive, but by using the augmented UKF form, non-additive noise may be handled as well. See the docstring of UnscentedKalmanFilter
for more details.
The UKF takes the same arguments as a regular KalmanFilter
, but the matrices defining the dynamics are replaced by two functions, dynamics
and measurement
, working in the same way as for the ParticleFilter
above (unless the augmented form is used).
ukf = UnscentedKalmanFilter(dynamics, measurement, cov(df), cov(dg), MvNormal(SA[1.,1.]); nu=nu, ny=ny)
UnscentedKalmanFilter{false, false, false, false, typeof(Main.dynamics), UKFMeasurementModel{false, false, typeof(Main.measurement), Matrix{Float64}, typeof(-), typeof(LowLevelParticleFilters.safe_mean), typeof(LowLevelParticleFilters.safe_cov), typeof(LowLevelParticleFilters.cross_cov), LowLevelParticleFilters.SigmaPointCache{Vector{StaticArraysCore.SVector{2, Float64}}, Vector{StaticArraysCore.SVector{1, Float64}}}}, Matrix{Float64}, Distributions.MvNormal{Float64, PDMats.PDiagMat{Float64, StaticArraysCore.SVector{2, Float64}}, FillArrays.Zeros{Float64, 1, Tuple{Base.OneTo{Int64}}}}, LowLevelParticleFilters.SigmaPointCache{Vector{StaticArraysCore.SVector{2, Float64}}, Vector{StaticArraysCore.SVector{2, Float64}}}, Vector{Float64}, Matrix{Float64}, LowLevelParticleFilters.NullParameters, Nothing, typeof(LowLevelParticleFilters.safe_mean), typeof(LowLevelParticleFilters.safe_cov), typeof(LinearAlgebra.cholesky!)}(Main.dynamics, UKFMeasurementModel{false, false, typeof(Main.measurement), Matrix{Float64}, typeof(-), typeof(LowLevelParticleFilters.safe_mean), typeof(LowLevelParticleFilters.safe_cov), typeof(LowLevelParticleFilters.cross_cov), LowLevelParticleFilters.SigmaPointCache{Vector{StaticArraysCore.SVector{2, Float64}}, Vector{StaticArraysCore.SVector{1, Float64}}}}(Main.measurement, [0.04000000000000001;;], 1, 0, -, LowLevelParticleFilters.safe_mean, LowLevelParticleFilters.safe_cov, LowLevelParticleFilters.cross_cov, LowLevelParticleFilters.SigmaPointCache{Vector{StaticArraysCore.SVector{2, Float64}}, Vector{StaticArraysCore.SVector{1, Float64}}}(StaticArraysCore.SVector{2, Float64}[[0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [0.0, 0.0]], StaticArraysCore.SVector{1, Float64}[[0.0], [0.0], [0.0], [0.0], [0.0]])), [0.010000000000000002 0.0; 0.0 0.010000000000000002], Distributions.MvNormal{Float64, PDMats.PDiagMat{Float64, StaticArraysCore.SVector{2, Float64}}, FillArrays.Zeros{Float64, 1, Tuple{Base.OneTo{Int64}}}}(
dim: 2
μ: Zeros(2)
Σ: [1.0 0.0; 0.0 1.0]
)
, LowLevelParticleFilters.SigmaPointCache{Vector{StaticArraysCore.SVector{2, Float64}}, Vector{StaticArraysCore.SVector{2, Float64}}}(StaticArraysCore.SVector{2, Float64}[[0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [0.0, 0.0]], StaticArraysCore.SVector{2, Float64}[[0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [0.0, 0.0]]), [0.0, 0.0], [1.0 0.0; 0.0 1.0], 0, 1.0, 1, 1, LowLevelParticleFilters.NullParameters(), nothing, LowLevelParticleFilters.safe_mean, LowLevelParticleFilters.safe_cov, LinearAlgebra.cholesky!)
If your function dynamics
describes a continuous-time ODE, do not forget to discretize it before passing it to the UKF. See Discretization for more information.
The UnscentedKalmanFilter
has many customization options, see the docstring for more details. In particular, the UKF may be created with a linear measurement model as an optimization.
Extended Kalman Filter
The ExtendedKalmanFilter
(EKF) is similar to the UKF, but propagates Gaussian distributions by linearizing the dynamics and using the formulas for linear systems similar to the standard Kalman filter. This can be slightly faster than the UKF (not always), but also less accurate for strongly nonlinear systems. The linearization is performed automatically using ForwardDiff.jl unless the user provides Jacobian functions that compute $A$ and $C$. In general, the UKF is recommended over the EKF unless the EKF is faster and computational performance is the top priority.
The EKF constructor has the following two signatures
ExtendedKalmanFilter(dynamics, measurement, R1, R2, d0=MvNormal(R1); nu::Int, p = LowLevelParticleFilters.NullParameters(), α = 1.0, check = true, Ajac = nothing, Cjac = nothing)
ExtendedKalmanFilter(kf, dynamics, measurement; Ajac = nothing, Cjac = nothing)
The first constructor takes all the arguments required to initialize the extended Kalman filter, while the second one takes an already defined standard Kalman filter. using the first constructor, the user must provide the number of inputs to the system, nu
.
where kf
is a standard KalmanFilter
from which the covariance properties are taken.
If your function dynamics
describes a continuous-time ODE, do not forget to discretize it before passing it to the UKF. See Discretization for more information.
AdvancedParticleFilter
The AdvancedParticleFilter
works very much like the ParticleFilter
, but admits more flexibility in its noise models.
The AdvancedParticleFilter
type requires you to implement the same functions as the regular ParticleFilter
, but in this case you also need to handle sampling from the noise distributions yourself. The function dynamics
must have a method signature like below. It must provide one method that accepts state vector, control vector, parameter, time and noise::Bool
that indicates whether or not to add noise to the state. If noise should be added, this should be done inside dynamics
An example is given below
using Random
const rng = Random.Xoshiro()
function dynamics(x, u, p, t, noise=false) # It's important that `noise` defaults to false
x = A*x .+ B*u # A simple linear dynamics model in discrete time
if noise
x += rand(rng, df) # it's faster to supply your own rng
end
x
end
The measurement_likelihood
function must have a method accepting state, input, measurement, parameter and time, and returning the log-likelihood of the measurement given the state, a simple example below:
function measurement_likelihood(x, u, y, p, t)
logpdf(dg, C*x-y) # An example of a simple linear measurement model with normal additive noise
end
This gives you very high flexibility. The noise model in either function can, for instance, be a function of the state, something that is not possible for the simple ParticleFilter
. To be able to simulate the AdvancedParticleFilter
like we did with the simple filter above, the measurement
method with the signature measurement(x,u,p,t,noise=false)
must be available and return a sample measurement given state (and possibly time). For our example measurement model above, this would look like this
# This function is only required for simulation
measurement(x, u, p, t, noise=false) = C*x + noise*rand(rng, dg)
We now create the AdvancedParticleFilter
and use it in the same way as the other filters:
apf = AdvancedParticleFilter(N, dynamics, measurement, measurement_likelihood, df, d0)
sol = forward_trajectory(apf, u, y, ny) # Perform batch filtering
ParticleFilteringSolution{AdvancedParticleFilter{PFstate{StaticArraysCore.SVector{2, Float64}, Float64}, typeof(Main.dynamics), typeof(Main.measurement), typeof(Main.measurement_likelihood), Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}, Distributions.IsoNormal, DataType, Random.Xoshiro, LowLevelParticleFilters.NullParameters}, Vector{StaticArraysCore.SVector{1, Float64}}, Vector{StaticArraysCore.SVector{1, Float64}}, Matrix{StaticArraysCore.SVector{2, Float64}}, Matrix{Float64}, Matrix{Float64}, Float64}(AdvancedParticleFilter{PFstate{StaticArraysCore.SVector{2, Float64}, Float64}, typeof(Main.dynamics), typeof(Main.measurement), typeof(Main.measurement_likelihood), Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}, Distributions.IsoNormal, DataType, Random.Xoshiro, LowLevelParticleFilters.NullParameters}
state: PFstate{StaticArraysCore.SVector{2, Float64}, Float64}
dynamics: dynamics (function of type typeof(Main.dynamics))
measurement: measurement (function of type typeof(Main.measurement))
measurement_likelihood: measurement_likelihood (function of type typeof(Main.measurement_likelihood))
dynamics_density: Distributions.ZeroMeanIsoNormal{Tuple{Base.OneTo{Int64}}}
initial_density: Distributions.IsoNormal
resample_threshold: Float64 0.5
resampling_strategy: LowLevelParticleFilters.ResampleSystematic <: LowLevelParticleFilters.ResamplingStrategy
rng: Random.Xoshiro
p: LowLevelParticleFilters.NullParameters LowLevelParticleFilters.NullParameters()
threads: Bool false
Ts: Float64 1.0
, StaticArraysCore.SVector{1, Float64}[[-0.36760465186560454], [0.2610360791501557], [-0.14025657421282536], [1.021039288329045], [0.24231240376406285], [-0.45127892754891835], [-0.3680669497405194], [0.7296076481683886], [1.8227795488942806], [-0.6165179242577173] … [-0.3196523941890061], [1.647811838294087], [2.2856369831616514], [1.9520878409962703], [-0.49818996628837353], [-0.4858789299822478], [2.4776391319996725], [1.2986635169117187], [-0.6898361546040195], [-1.5812595201552577]], StaticArraysCore.SVector{1, Float64}[[-0.17291311138147836], [-0.17078198268360928], [-0.016807954215372685], [0.16865612155111984], [-0.4578238432646786], [0.3082616887520184], [0.09146709696928511], [-0.1892980366461109], [0.06962279285190669], [0.28136022045378567] … [0.8562035034255762], [0.5057965220960559], [0.4588229822986169], [0.4930460472890044], [0.26124319727817075], [-0.01793794946321764], [0.5479396660927862], [0.38888921972743085], [0.7808401345244972], [0.6896564363157827]], StaticArraysCore.SVector{2, Float64}[[0.19469120711559784, -0.6882466101224325] [-0.9543080376672683, 0.03617481045331411] … [0.08625936032404347, 0.3126374178101856] [0.5097851826840722, 0.5274900701930858]; [1.4694748882539674, 0.8541022231264515] [2.118318929410887, 0.19749606526550662] … [0.5916460228071517, 0.3842244395535429] [-0.04272291070509053, 0.5081398172707049]; … ; [1.4891231622594312, 2.032890502137377] [-2.157693008100583, -0.5276357541725223] … [0.8670811336574319, 0.5188791245188776] [0.8250850164912898, 0.4920811076779563]; [-1.6841376651820785, -3.8285227996902753] [-2.1619434540066025, -0.555817752850289] … [0.8622758756907233, 0.43891247576169623] [0.529003364340642, 0.4759399431882674]], [-8.677827376088274 -7.529164626203818 … -8.552929875937883 -7.447162547178668; -18.542725905961298 -8.689134705899967 … -7.779057695420773 -7.530291955780109; … ; -66.17783945719137 -8.585583376208096 … -6.670552211244164 -7.606388549489974; -172.40174710524505 -8.846932502232928 … -7.274189124542241 -7.689372661344702], [0.00017032070875251947 0.000537186821093889 … 0.00019297886632961282 0.0005830937681855768; 8.851083943267644e-9 0.00016840568363903552 … 0.00041840625504005805 0.0005365815757220791; … ; 1.8168936181675832e-29 0.00018677920209147963 … 0.0012676985203102747 0.0004972644655809797; 1.3392836337166474e-75 0.0001438222347007188 … 0.0006932019933553252 0.0004576651927625608], -11.775372629411233)
plot(sol, xreal=x)
We can even use this type as an AuxiliaryParticleFilter
apfa = AuxiliaryParticleFilter(apf)
sol = forward_trajectory(apfa, u, y, ny)
plot(sol, dim=1, xreal=x) # Same as above, but only plots a single dimension
See the tutorials section for more advanced examples, including state estimation for DAE (Differential-Algebraic Equation) systems.
Troubleshooting and tuning
Tuning a particle filter can be quite the challenge. To assist with this, we provide som visualization tools
debugplot(pf,u[1:20],y[1:20], runall=true, xreal=x[1:20])
Time Surviving Effective nbr of particles
--------------------------------------------------------------
t: 1 1.000 2000.0
t: 2 1.000 280.0
t: 3 0.150 2000.0
t: 4 1.000 1294.6
t: 5 1.000 781.2
t: 6 1.000 309.4
t: 7 1.000 253.0
t: 8 0.233 2000.0
t: 9 1.000 1038.7
t: 10 1.000 948.0
t: 11 1.000 759.3
t: 12 1.000 606.2
t: 13 1.000 510.8
t: 14 1.000 422.3
t: 15 1.000 341.1
t: 16 1.000 252.0
t: 17 1.000 213.7
t: 18 0.212 2000.0
t: 19 1.000 1468.2
t: 20 1.000 1218.4
The plot displays all state variables and all measurements. The heatmap in the background represents the weighted particle distributions per time step. For the measurement sequences, the heatmap represent the distributions of predicted measurements. The blue dots corresponds to measured values. In this case, we simulated the data and we had access to the state as well, if we do not have that, just omit xreal
. You can also manually step through the time-series using
commandplot(pf,u,y; kwargs...)
For options to the debug plots, see ?pplot
.
Tuning noise parameters through optimization
See examples in Parameter Estimation.
Tuning through simulation
It is possible to sample from the Bayesian model implied by a filter and its parameters by calling the function simulate
. A simple tuning strategy is to adjust the noise parameters such that a simulation looks "similar" to the data, i.e., the data must not be too unlikely under the model.
Videos
Several video tutorials using this package are available in the playlists
Some examples featuring this package in particular are
Using an optimizer to optimize the likelihood of an UnscentedKalmanFilter
:
Estimation of time-varying parameters:
Adaptive control by means of estimation of time-varying parameters: