LGCPs - Multiple Likelihoods
Fabian E. Bachl
Generated on 2024-09-11
Source:vignettes/articles/2d_lgcp_multilikelihood.Rmd
2d_lgcp_multilikelihood.Rmd
Introduction
For this vignette we are going to be working with the inlabru’s
´gorillas_sf´ dataset which was originally obtained from the
R
package spatstat
. The data set contains two
types of gorillas nests which are marked as either major or minor. We
will set up a multi-likelihood model for these nests which creates two
spatial LGCPs that share a common intercept but have employ different
spatial smoothers.
Get the data
For the next few practicals we are going to be working with a dataset
obtained from the R
package spatstat
, which
contains the locations of 647 gorilla nests. We load the dataset like
this:
data(gorillas_sf, package = "inlabru")
Plot the nests and visualize the group membership (major/minor) by color:
Fiting the model
First, we define all components that enter the joint model. That is, the intercept that is common to both LGCPs and the two different spatial smoothers, one for each nest group.
matern <- inla.spde2.pcmatern(gorillas_sf$mesh,
prior.range = c(0.1, 0.01),
prior.sigma = c(1, 0.01)
)
cmp <- ~
Common(geometry, model = matern) +
Difference(geometry, model = matern) +
Intercept(1)
Given these components we define the linear predictor for each of the
likelihoods. (Using “.” indicates a pure additive model, and one can use
include/exclude options for like()
to indicate which
components are actively involved in each model.)
fml.major <- geometry ~ Intercept + Common + Difference / 2
fml.minor <- geometry ~ Intercept + Common - Difference / 2
Setting up the Cox process likelihoods is easy in this example. Both nest types were observed within the same window:
lik_minor <- like("cp",
formula = fml.major,
data = gorillas_sf$nests[gorillas_sf$nests$group == "major", ],
samplers = gorillas_sf$boundary,
domain = list(geometry = gorillas_sf$mesh)
)
lik_major <- like("cp",
formula = fml.minor,
data = gorillas_sf$nests[gorillas_sf$nests$group == "minor", ],
samplers = gorillas_sf$boundary,
domain = list(geometry = gorillas_sf$mesh)
)
… which we provide to the ´bru´ function.
jfit <- bru(cmp, lik_major, lik_minor,
options = list(
control.inla = list(int.strategy = "eb"),
bru_max_iter = 1
)
)
library(patchwork)
pl.major <- ggplot() +
gg(gorillas_sf$mesh,
mask = gorillas_sf$boundary,
col = exp(jfit$summary.random$Common$mean)
)
pl.minor <- ggplot() +
gg(gorillas_sf$mesh,
mask = gorillas_sf$boundary,
col = exp(jfit$summary.random$Difference$mean)
)
(pl.major + scale_fill_continuous(trans = "log")) +
(pl.minor + scale_fill_continuous(trans = "log")) &
theme(legend.position = "right")
Rerunning
Rerunning with the previous estimate as starting point sometimes improves the accuracy of the posterior distribution estimation.
jfit0 <- jfit
jfit <- bru_rerun(jfit)
library(patchwork)
pl.major <- ggplot() +
gg(gorillas_sf$mesh,
mask = gorillas_sf$boundary,
col = exp(jfit$summary.random$Common$mean)
)
pl.minor <- ggplot() +
gg(gorillas_sf$mesh,
mask = gorillas_sf$boundary,
col = exp(jfit$summary.random$Difference$mean)
)
(pl.major + scale_fill_continuous(trans = "log")) +
(pl.minor + scale_fill_continuous(trans = "log")) &
theme(legend.position = "right")
summary(jfit0)
#> inlabru version: 2.11.1.9005
#> INLA version: 24.06.27
#> Components:
#> Common: main = spde(geometry), group = exchangeable(1L), replicate = iid(1L), NULL
#> Difference: main = spde(geometry), group = exchangeable(1L), replicate = iid(1L), NULL
#> Intercept: main = linear(1), group = exchangeable(1L), replicate = iid(1L), NULL
#> Likelihoods:
#> Family: 'cp'
#> Data class: 'sf', 'data.frame'
#> Response class: 'numeric'
#> Predictor: geometry ~ Intercept + Common - Difference/2
#> Used components: effects[Common, Difference, Intercept], latent[]
#> Family: 'cp'
#> Data class: 'sf', 'data.frame'
#> Response class: 'numeric'
#> Predictor: geometry ~ Intercept + Common + Difference/2
#> Used components: effects[Common, Difference, Intercept], latent[]
#> Time used:
#> Pre = 0.646, Running = 20.4, Post = 0.157, Total = 21.2
#> Fixed effects:
#> mean sd 0.025quant 0.5quant 0.975quant mode kld
#> Intercept -0.356 1.381 -3.064 -0.356 2.351 -0.356 0
#>
#> Random effects:
#> Name Model
#> Common SPDE2 model
#> Difference SPDE2 model
#>
#> Model hyperparameters:
#> mean sd 0.025quant 0.5quant 0.975quant mode
#> Range for Common 2.937 0.573 1.990 2.876 4.24 2.748
#> Stdev for Common 2.045 0.333 1.479 2.015 2.79 1.950
#> Range for Difference 2.247 3.096 0.150 1.309 10.10 0.398
#> Stdev for Difference 0.161 0.090 0.038 0.143 0.38 0.102
#>
#> Deviance Information Criterion (DIC) ...............: 590.48
#> Deviance Information Criterion (DIC, saturated) ....: 588.84
#> Effective number of parameters .....................: -773.63
#>
#> Watanabe-Akaike information criterion (WAIC) ...: 1629.02
#> Effective number of parameters .................: 104.95
#>
#> Marginal log-Likelihood: -1204.08
#> is computed
#> Posterior summaries for the linear predictor and the fitted values are computed
#> (Posterior marginals needs also 'control.compute=list(return.marginals.predictor=TRUE)')
summary(jfit)
#> inlabru version: 2.11.1.9005
#> INLA version: 24.06.27
#> Components:
#> Common: main = spde(geometry), group = exchangeable(1L), replicate = iid(1L), NULL
#> Difference: main = spde(geometry), group = exchangeable(1L), replicate = iid(1L), NULL
#> Intercept: main = linear(1), group = exchangeable(1L), replicate = iid(1L), NULL
#> Likelihoods:
#> Family: 'cp'
#> Data class: 'sf', 'data.frame'
#> Response class: 'numeric'
#> Predictor: geometry ~ Intercept + Common - Difference/2
#> Used components: effects[Common, Difference, Intercept], latent[]
#> Family: 'cp'
#> Data class: 'sf', 'data.frame'
#> Response class: 'numeric'
#> Predictor: geometry ~ Intercept + Common + Difference/2
#> Used components: effects[Common, Difference, Intercept], latent[]
#> Time used:
#> Pre = 0.52, Running = 7.83, Post = 0.131, Total = 8.48
#> Fixed effects:
#> mean sd 0.025quant 0.5quant 0.975quant mode kld
#> Intercept -0.353 1.366 -3.031 -0.353 2.325 -0.353 0
#>
#> Random effects:
#> Name Model
#> Common SPDE2 model
#> Difference SPDE2 model
#>
#> Model hyperparameters:
#> mean sd 0.025quant 0.5quant 0.975quant mode
#> Range for Common 2.936 0.573 2.000 2.872 4.245 2.733
#> Stdev for Common 2.046 0.332 1.486 2.015 2.791 1.944
#> Range for Difference 2.144 3.010 0.134 1.231 9.754 0.356
#> Stdev for Difference 0.159 0.095 0.036 0.139 0.393 0.096
#>
#> Deviance Information Criterion (DIC) ...............: 590.51
#> Deviance Information Criterion (DIC, saturated) ....: 588.87
#> Effective number of parameters .....................: -773.65
#>
#> Watanabe-Akaike information criterion (WAIC) ...: 1629.04
#> Effective number of parameters .................: 104.93
#>
#> Marginal log-Likelihood: -1204.01
#> is computed
#> Posterior summaries for the linear predictor and the fitted values are computed
#> (Posterior marginals needs also 'control.compute=list(return.marginals.predictor=TRUE)')
Single-likelihood version
In this particular model, we can also rewrite the problem as a single
point process over a product domain over space and group
.
In versions <= 2.7.0
, the integration domain had to be
numeric, so we convert the group variable to a 0/1 variable,
group_major <- group == "major"
, which is also useful in
the predictor expression:
fml.joint <- geometry + group_major ~ Intercept + Common + (group_major - 0.5) * Difference
gorillas_sf$nests$group_major <- gorillas_sf$nests$group == "major"
lik_joint <- like("cp",
formula = fml.joint,
data = gorillas_sf$nests,
samplers = gorillas_sf$boundary,
domain = list(
geometry = gorillas_sf$mesh,
group_major = c(0, 1)
)
)
jfit_joint <- bru(cmp, lik_joint,
options = list(
control.inla = list(int.strategy = "eb") # Approximate for faster vignette
)
)
Plotting the ratios of exp(Common)
and
exp(Difference)
between the new fit and the old confirms
that the results are the same up to small numerical differences.
library(patchwork)
pl.major <- ggplot() +
gg(gorillas_sf$mesh,
mask = gorillas_sf$boundary,
col = exp(jfit_joint$summary.random$Common$mean -
jfit$summary.random$Common$mean)
)
pl.minor <- ggplot() +
gg(gorillas_sf$mesh,
mask = gorillas_sf$boundary,
col = exp(jfit_joint$summary.random$Difference$mean -
jfit$summary.random$Difference$mean)
)
(pl.major + scale_fill_continuous(trans = "log")) +
(pl.minor + scale_fill_continuous(trans = "log")) &
theme(legend.position = "right")