Handling Particle Depletion
Many of the most common problems with particle filters are related to particle depletion, that is, a lack of particles corresponding to the true state. In many cases, it is not difficult to overcome these problems, but domain-specific heuristics are often more effective than generic approaches.
The recommended way to handle particle depletion in a BootstrapFilter
or BasicParticleFilter
is to add a postprocessing step. In the BasicParticleFilter
, this is a required argument; in the BootstrapFilter
, it is an optional keyword argument.
A postprocessing function takes 6 arguments:
postprocess(bp, a, o, b, bb, rng)
bp
is the new belief, a
is the action, o
is the observation, b
is the old belief, bb
is the belief after preprocessing, and rng
is the random number generator. See the Basic Particle Filter section to see where postprocess
is called in the code. The postprocessing function should return the modified belief. This can be the same object bp
modified in place or a new object.
If you don't need all of the arguments, you can use the splat operator ...
to ignore them, e.g. postprocess(bp, a, o, args...)
.
The function does not need to be named postprocess
. It can have any name or be anonymous.
Creating a postprocessing function is best illustrated by a simple example:
Example
In this example, we have a system with integer states states and actions, deterministic dynamics, and an observation that is uniformly distributed over the current state plus or minus 1. If all state particles start 1, action 1 is taken, and observation 4 is received, then no particles are consistent with the observation. This will yield a warning about particle depletion and produce a belief with zero-weighted particles that cannot be sampled from or used effectively:
using ParticleFilters, Distributions
dynamics(s, a, rng) = s + a
likelihood(s_previous, a, s, o) = abs(o - s) <= 1
naive_pf = BootstrapFilter(dynamics, likelihood, 3)
b0 = ParticleCollection([1, 1, 1])
a = 1
o = 4
bp = update(naive_pf, b0, a, o)
┌ Warning: Sum of particle filter weights is not greater than zero. This
│ could be due to particle depletion. See the documentation of the
│ ParticleFilters package for instructions on handling particle depletion.
│ weight_sum(b) = 0.0
└ @ ParticleFilters ~/work/ParticleFilters.jl/ParticleFilters.jl/src/postprocessing.jl:8
To fix this, we define a postprocessing step to refill the particle belief with particles consistent with the observation:
function refill_with_consistent(bp, a, o, b, bb, rng)
if weight_sum(bp) == 0.0
return WeightedParticleBelief([o-1, o, o+1], ones(3))
else
return bp
end
end
Note that this solution is very domain-specific. It relies on specific knowledge about the observation function. In other applications, you will likely need to be clever about coming up with ways to create replacement particles in cases of depletion. Armed with the new postprocessing step, we can create a new filter and use it for a belief update:
pf = BootstrapFilter(dynamics, likelihood, 3; postprocess=refill_with_consistent)
bp = update(pf, b0, a, o)
WeightedParticleBelief{Int64}([3, 4, 5], [1.0, 1.0, 1.0], 3.0, nothing, nothing)
While this new belief is consistent with the observation, it does not take into account the dynamics or information from the previous steps. A better solution might be to populate the new belief with particles as near as possible to the previous particles while still being consistent with the observation. Tricks like this are often necessary to get good performance in practice and they are an important part of the art of particle filtering.
Additional Examples of Postprocessing Functions
The following section provides additional examples of strategies that commonly work well and additional postprocessing functions.
Replacing Zero-Weight Particles
Rather than waiting until all particle weights are zero, it is often effective to replace zero-weight particles as soon as they are detected. If a function consistent_state(o, rng)
is written to generate states consistent with the observation, this replacement can be accomplished with the following postprocessing function:
function replace_zero_weight_particles(bp, a, o, b, bb, rng)
n = n_particles(bp)
new_weight = weight_sum(bp)/n
for i in 1:n
if weight(bp, i) == 0.0
s = consistent_state(o, rng)
set_pair!(bp, i, s => new_weight)
end
end
return bp
end
Adding Artificial Noise
In continuous-state systems with deterministic dynamics, it can be useful to add artificial noise to the states as follows:
function add_noise(bp, a, o, b, bb, rng)
for i in 1:n_particles(bp)
p = particle(bp, i)
set_particle!(bp, i, p + 0.1*randn(rng))
end
return bp
end
Combining Multiple Postprocessing Steps
It is often useful to combine multiple postprocessing steps. This can be accomplished convenienttly with the PostprocessChain
function:
PostprocessChain(replace_zero_weight_particles, add_noise)
Replacing Zero-Weight Particles with States from the Initial Distribution
If no other information is available, a quick solution is to replace zero-weight particles with states drawn from the initial distribution. In POMDPs.jl, the initialstate
function can be used to generate these states. This implementation uses a callable object to store the POMDP model.
using ParticleFilters, POMDPs, POMDPModels
struct ReplaceWithInitial{M <: POMDP} <: Function
m::M
threshold::Float64 # replace particles with normalized weights below this
end
function (r::ReplaceWithInitial)(bp, a, o, b, bb, rng)
ws = weights(bp)
wsum = weight_sum(bp)
nws = ws ./ wsum
n = n_particles(bp)
for i in 1:n
if nws[i] < r.threshold
set_pair!(bp, i, rand(rng, initial_state(r.m)) => wsum/n)
end
end
return bp
end
The filter can then be created as follows:
m = TigerPOMDP()
pf = BootstrapFilter(m, 1, postprocess=ReplaceWithInitial(m, 0.1))
BasicParticleFilter{ParticleFilters.NormalizedESSConditionalResampler{typeof(low_variance_sample)}, ParticleFilters.POMDPPredictor{POMDPModels.TigerPOMDP}, ParticleFilters.POMDPReweighter{POMDPModels.TigerPOMDP}, ParticleFilters.var"#21#23"{Main.ReplaceWithInitial{POMDPModels.TigerPOMDP}, typeof(check_particle_belief)}, ParticleFilters.var"#25#28"{Int64}, Random.TaskLocalRNG}(ParticleFilters.NormalizedESSConditionalResampler{typeof(low_variance_sample)}(ParticleFilters.low_variance_sample, 1, 0.9), ParticleFilters.POMDPPredictor{POMDPModels.TigerPOMDP}(POMDPModels.TigerPOMDP(-1.0, -100.0, 10.0, 0.85, 0.95)), ParticleFilters.POMDPReweighter{POMDPModels.TigerPOMDP}(POMDPModels.TigerPOMDP(-1.0, -100.0, 10.0, 0.85, 0.95)), ParticleFilters.var"#21#23"{Main.ReplaceWithInitial{POMDPModels.TigerPOMDP}, typeof(check_particle_belief)}(Main.ReplaceWithInitial{POMDPModels.TigerPOMDP}(POMDPModels.TigerPOMDP(-1.0, -100.0, 10.0, 0.85, 0.95), 0.1), ParticleFilters.check_particle_belief), ParticleFilters.var"#25#28"{Int64}(1), Random.TaskLocalRNG())