Direct maximization of the particle filter likelihood
Produced in R version 4.3.2.
In the toy example we’ve been working with, the default parameter set is not particularly close to the MLE. One way to find the MLE is to try optimizing the estimated likelihood directly. There are of course many standard optimization algorithms we might use for this. However, three issues arise immediately:
Let’s try this out on the toy SIR model we were working with, reconstructed as follows.
source("https://kingaa.github.io/sbied/pfilter/model.R")
Here, let’s opt for deterministic optimization of a rough function. We’ll try using optim
’s default method: Nelder-Mead, fixing the random-number generator seed to make the likelihood calculation deterministic. Since Nelder-Mead is an unconstrained optimizer, we must transform the parameters. The following introduces such a transformation into the pomp
object.
measSIR |>
pomp(partrans=parameter_trans(log=c("Beta","mu_IR"),logit=c("rho","eta")),
paramnames=c("Beta","mu_IR","eta","rho")) -> measSIR
We can think of the parameters that we furnished when creating measSIR
as a kind of reference point in parameter space.
coef(measSIR)
Beta mu_IR rho k eta N
1.5e+01 5.0e-01 5.0e-01 1.0e+01 6.0e-02 3.8e+04
The following constructs a function returning the negative log likelihood of the data at a given point in parameter space. The parameters to be estimated are named in the est
argument; the others will remain fixed at the reference values. Note how the freeze
function is used to fix the seed of the RNG. Note too, how this function returns a large (and therefore bad) value when the particle filter encounters and error. This behavior makes the objective function more robust.
neg.ll <- function (par, est) {
try(
freeze({
allpars <- coef(measSIR,transform=TRUE)
allpars[est] <- par
theta <- partrans(measSIR,allpars,dir="fromEst")
pfilter(measSIR,params=theta,Np=2000)
},
seed=915909831
)
) -> pf
if (inherits(pf,"try-error")) 1e10 else -logLik(pf)
}
Now we call optim
to minimize this function:
## use Nelder-Mead with fixed RNG seed
estpars <- c("Beta","mu_IR","eta")
optim(
par=coef(measSIR,estpars,transform=TRUE),
est=estpars,
fn=neg.ll,
method="Nelder-Mead",
control=list(maxit=400,trace=0)
) -> fit
mle <- measSIR
coef(mle,estpars,transform=TRUE) <- fit$par
coef(mle)
fit$val
lls <- replicate(n=5,logLik(pfilter(mle,Np=20000)))
ll <- logmeanexp(lls,se=TRUE); ll
We plot some simulations at these parameters.
mle |> simulate(nsim=10,format="data.frame",include.data=TRUE) -> sims
The data are shown in blue. The 10 simulations are shown in red.
The search of parameter space we conducted above was local. It is possible that we found a local maximum, but that other maxima exist with higher likelihoods. Conduct a more thorough search by initializing the Nelder-Mead starting points across a wider region of parameter space. Do you find any other local maxima?
Try to estimate β, μIR, ρ, and η simultaneously. Does your estimate of R0 differ from the value we computed from the raw data? How do you interpret the agreement or lack thereof?
Licensed under the Creative Commons Attribution-NonCommercial license. Please share and remix noncommercially, mentioning its origin.