(hrue@r-inla.org)
conditional-sampling.RmdDetails of this vignette requires the classic-mode, but it is possible to make it work for the new compact-mode. Please contact if that is required.
inla.setOption(inla.mode="classic")This short note describe how to do conditional sampling from a fitted model, where we want to condition on new events. An easy example which motivated this note, is the following.
n = 50
grp.len = 5
x = rnorm(n)
g = rnorm(grp.len)
grp = rep(1:grp.len, each = n %/% grp.len)
y = 1 + x + g[grp] + rnorm(n, sd = 0.01)and the task is to provide the posterior a new observation, condition on that the random effect āgā is zero. Although it is possible to get this directly (an exercise for the reader), in general, we will show that it is easier to compute this using Monte Carlo sampling and the function . With a little hack.
To fit the model, we do
r = inla(y ~ x + f(grp, model = "iid"),
data = data.frame(y, x, grp),
control.compute = list(config=TRUE)) adding to prepare for the use of .
The trick, is to add the conditioning as constraints, like
where
is the latent field. In our case, it is simply t grp.len
constraints,
for
from 1 to 5. To do this, we need to create the matrix
,
and vector
,
and add it to the list of contraints (if any). To do this, we need to
know the index of the -term, which we find here
r$misc$configs$contents## $tag
## [1] "Predictor" "grp" "(Intercept)" "x"
##
## $start
## [1] 1 51 56 57
##
## $length
## [1] 50 5 1 1
so we have both the start index and the length of this vector. So we create a constraint to represent constraints, for from 1 to 5.
m = sum(r$misc$configs$contents$length)
grp.idx = which(r$misc$configs$contents$tag == "grp")
grp.len = r$misc$configs$contents$length[grp.idx]
A = matrix(0, grp.len, m)
e = matrix(0, grp.len, 1)
for(i in 1:grp.len) {
A[i, r$misc$configs$contents$start[grp.idx] + i - 1] = 1
e[i] = 0
}Then we need to append it to the existing one if any, or to add a new one.
constr = r$misc$configs$constr
if (is.null(constr)) {
## nothing there from before
r$misc$configs$constr = list(
nc = dim(A)[2],
A = A,
e = e)
} else {
## create a new one
r$misc$configs$constr = list(
nc = constr$nc + dim(A)[2],
A = rbind(constr$A, A),
e = rbind(constr$e, e))
}When we now do
xx = inla.posterior.sample(1000, r)we additional condition on , but would still appear in the list of samples as zero or something very close to it. The rest should be straight forward.
PS: Note that you can add any linear combination of as a constraint, you just do not want to many, as the cost is quadratic the number of constraints. If the constraints simply set variables to zero, you can also add a large number to , which you will find in , one for each configuration.