-
Notifications
You must be signed in to change notification settings - Fork 89
/
Copy path16-SpatialRegression.qmd
426 lines (340 loc) · 24.1 KB
/
16-SpatialRegression.qmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
# Spatial Regression {#sec-spatglmm}
```{r echo = FALSE}
eval_inla = Sys.getenv("EVAL_INLA") != "false"
```
\index{spatial regression}
\index{regression!spatial}
Even though it may be tempting to focus on interpreting the map pattern of an areal support response variable of interest, the pattern may largely derive from covariates (and their functional forms), as well as the respective spatial footprints of the variables in play. Spatial autoregressive models in two dimensions began without covariates and with clear links to time series [@whittle:54]. Extensions included tests for spatial autocorrelation in linear model residuals, and models applying the autoregressive component to the response or the residuals, where the latter matched the tests for residuals [@CliffOrd:72; @cliff+ord:73]. These "lattice" models of areal data typically express the dependence between observations using a graph of neighbours in the form of a contiguity matrix.
\index{regression!generalised least squares}
\index{linear model!mixed effects}
Of course, handling a spatial correlation structure in a generalised least squares model or a (generalised) linear or non-linear mixed effects model such as those provided in the **nlme** and many other packages does not have to use a graph of neighbours [@R:Pinheiro+Bates:2000]. These models are also spatial regression models, using functions of the distance between observations, and fitted variograms to model the spatial autocorrelation present; such models have been held to yield a clearer picture of the underlying processes [@wall:04], building on geostatistics. For example, the **glmmTMB** package successfully uses this approach to spatial regression [@brookesetal:17]. Here we will only consider spatial regression using spatial weights matrices.
## Markov random field and multilevel models
\index{Markov random field}
\index{multilevel model}
\index{linear model!multilevel}
\index{linear model!conditional autoregressive}
\index{linear model!intrinsic CAR}
\index{conditional autoregressive model}
\index{intrinsic CAR model}
There is a large literature in disease mapping using conditional autoregressive (CAR) and intrinsic CAR (ICAR) models in spatially structured random effects. These extend to multilevel models, in which the spatially structured random effects may apply at different levels of the model [@bivandetal17a]. In order to try out some of the variants, we need to remove the no-neighbour observations from the tract level, and from the model output zone aggregated level, in two steps as reducing the tract level induces a no-neighbour outcome at the model output zone level. Many of the model estimating functions take `family` arguments, and fit generalised linear mixed effects models with per-observation spatial random effects structured using a Markov random field representation of relationships between neighbours. In the multilevel case, the random effects may be modelled at the group level, which is the case presented in the following examples.
We follow @vgr_clubuc3m:19 in summarising @R:Pinheiro+Bates:2000 and @mcculloch+searle:2001 to describe the mixed-effects model representation of spatial regression models. In a Gaussian linear mixed model setting, a random effect $u$ is added to the model, with response $Y$, fixed covariates $X$, their coefficients $\beta$ and error term $\varepsilon_i \sim N(0, \sigma^2), i=1,\dots, n$:
$$
Y = X \beta + Z u + \varepsilon
$$
$Z$ is a fixed design matrix for the random effects. If there are $n$ random effects, it will be an $n \times n$ identity matrix if instead the observations are aggregated into $m$ groups, so with $m < n$ random effects, it will be an $n \times m$ matrix showing which group each observation belongs to. The random effects are modelled as a multivariate Normal distribution $u \sim N(0, \sigma^2_u \Sigma)$, and $\sigma^2_u \Sigma$ is the square variance-covariance matrix of the random effects.
\index{SAR models}
\index{CAR models}
\index{linear model!SAR, CAR}
A division has grown up, possibly unhelpfully, between scientific fields using CAR models [@besag:74], and simultaneous autoregressive models (SAR) [@ord:75; @hepple:76]. Although CAR and SAR models are closely related, these fields have found it difficult to share experience of applying similar models, often despite referring to key work summarising the models [@ripley:81; @ripley:88; @Cressie:1993]. Ripley gives the SAR variance as [@ripley:81, page 89], here shown as the inverse $\Sigma^{-1}$ (also known as the precision matrix):
$$
\Sigma^{-1} = [(I - \rho W)'(I - \rho W)]
$$
where $\rho$ is a spatial autocorrelation parameter and $W$ is a non-singular spatial weights matrix that represents spatial dependence. The CAR variance is:
$$
\Sigma^{-1} = (I - \rho W)
$$
where $W$ is a symmetric and strictly positive definite spatial weights matrix. In the case of the intrinsic CAR model, avoiding the estimation of a spatial autocorrelation parameter, we have:
$$
\Sigma^{-1} = M = \mathrm{diag}(n_i) - W
$$
where $W$ is a symmetric and strictly positive definite spatial weights matrix as before and $n_i$ are the row sums of $W$. The Besag-York-Mollié model includes intrinsic CAR spatially structured random effects and unstructured random effects. The Leroux model combines matrix components for unstructured and spatially structured random effects, where the spatially structured random effects are taken as following an intrinsic CAR specification:
$$
\Sigma^{-1} = [(1 - \rho) I_n + \rho M]
$$
References to the definitions of these models may be found in @gómez2020bayesian, and estimation issues affecting the Besag-York-Mollié and Leroux models are reviewed by @JSSv063c01.
More recent books expounding the theoretical bases for modelling with areal data simply point out the similarities between SAR and CAR models in relevant chapters [@gaetan+guyon:10; @vanlieshout:19]; the interested reader is invited to consult these sources for background information.
### Boston house value dataset
Here we shall use the Boston housing dataset, which has been restructured and furnished with census tract boundaries [@bivand17]. The original dataset used 506 census tracts and a hedonic model to try to estimate willingness to pay for clean air. The response was constructed from counts of ordinal answers to a 1970 census question about house value. The response is left- and right-censored in the census source and has been treated as Gaussian. The key covariate was created from a calibrated meteorological model showing the annual nitrogen oxides (NOX) level for a smaller number of model output zones. The numbers of houses responding also varies by tract and model output zone. There are several other covariates, some measured at the tract level, some by town only, where towns broadly correspond to the air pollution model output zones.
```{r setup_sr0, include=FALSE}
knitr::opts_chunk$set(echo = TRUE, paged.print = FALSE)
```
We can start by reading in the 506 tract dataset from **spData** [@R-spData], and creating a contiguity neighbour object and from that again a row standardised spatial weights object.
```{r, message=FALSE}
library(sf)
library(spData)
boston_506 <- st_read(system.file("shapes/boston_tracts.gpkg",
package = "spData")[1], quiet = TRUE)
```
```{r}
nb_q <- spdep::poly2nb(boston_506)
lw_q <- spdep::nb2listw(nb_q, style = "W")
```
If we examine the median house values, we find that those for censored values have been assigned as missing, and that 17 tracts are affected.
```{r}
table(boston_506$censored)
```
```{r}
summary(boston_506$median)
```
Next, we can subset to the remaining 489 tracts with non-censored house values, and the neighbour object to match. The neighbour object now has one observation with no neighbours.
```{r}
boston_506$CHAS <- as.factor(boston_506$CHAS)
boston_489 <- boston_506[!is.na(boston_506$median),]
nb_q_489 <- spdep::poly2nb(boston_489)
lw_q_489 <- spdep::nb2listw(nb_q_489, style = "W",
zero.policy = TRUE)
```
The `NOX_ID` variable specifies the upper-level aggregation, letting us aggregate the tracts to air pollution model output zones. We can create aggregate neighbour and row standardised spatial weights objects, and aggregate the `NOX` variable taking means, and the `CHAS` Charles River dummy variable for observations on the river. Here we follow the principles outlined in @sec-extensiveintensive for spatially extensive and intensive variables; neither `NOX` nor `CHAS` can be summed, as they are not count variables.
```{r}
agg_96 <- list(as.character(boston_506$NOX_ID))
boston_96 <- aggregate(boston_506[, "NOX_ID"], by = agg_96,
unique)
nb_q_96 <- spdep::poly2nb(boston_96)
lw_q_96 <- spdep::nb2listw(nb_q_96)
boston_96$NOX <- aggregate(boston_506$NOX, agg_96, mean)$x
boston_96$CHAS <-
aggregate(as.integer(boston_506$CHAS)-1, agg_96, max)$x
```
The response is aggregated using the `weightedMedian` function in **matrixStats**, and midpoint values for the house value classes. Counts of houses by value class were punched to check the published census values, which can be replicated using `weightedMedian` at the tract level. Here we find two output zones with calculated weighted medians over the upper census question limit of USD \$50,000, and remove them subsequently as they also are affected by not knowing the appropriate value to insert for the top class by value. This is a case of spatially extensive aggregation, for which the summation of counts is appropriate:
```{r}
nms <- names(boston_506)
ccounts <- 23:31
for (nm in nms[c(22, ccounts, 36)]) {
boston_96[[nm]] <- aggregate(boston_506[[nm]], agg_96, sum)$x
}
br2 <-
c(3.50, 6.25, 8.75, 12.5, 17.5, 22.5, 30, 42.5, 60) * 1000
counts <- as.data.frame(boston_96)[, nms[ccounts]]
f <- function(x) matrixStats::weightedMedian(x = br2, w = x,
interpolate = TRUE)
boston_96$median <- apply(counts, 1, f)
is.na(boston_96$median) <- boston_96$median > 50000
summary(boston_96$median)
```
Before subsetting, we aggregate the remaining covariates by weighted mean using the tract population counts punched from the census [@bivand17]; these are spatially intensive variables, not count data.
```{r}
POP <- boston_506$POP
f <- function(x) matrixStats::weightedMean(x[,1], x[,2])
for (nm in nms[c(9:11, 14:19, 21, 33)]) {
s0 <- split(data.frame(boston_506[[nm]], POP), agg_96)
boston_96[[nm]] <- sapply(s0, f)
}
boston_94 <- boston_96[!is.na(boston_96$median),]
nb_q_94 <- spdep::subset.nb(nb_q_96, !is.na(boston_96$median))
lw_q_94 <- spdep::nb2listw(nb_q_94, style="W")
```
We now have two datasets at each level, at the lower, census tract level, and at the upper, air pollution model output zone level, one including the censored observations, the other excluding them.
```{r}
boston_94a <- aggregate(boston_489[,"NOX_ID"],
list(boston_489$NOX_ID), unique)
nb_q_94a <- spdep::poly2nb(boston_94a)
NOX_ID_no_neighs <-
boston_94a$NOX_ID[which(spdep::card(nb_q_94a) == 0)]
boston_487 <- boston_489[is.na(match(boston_489$NOX_ID,
NOX_ID_no_neighs)),]
boston_93 <- aggregate(boston_487[, "NOX_ID"],
list(ids = boston_487$NOX_ID), unique)
row.names(boston_93) <- as.character(boston_93$NOX_ID)
nb_q_93 <- spdep::poly2nb(boston_93,
row.names = unique(as.character(boston_93$NOX_ID)))
```
\newpage
The original model related the log of median house values by tract to the square of NOX values, including other covariates usually related to house value by tract, such as aggregate room counts, aggregate age, ethnicity, social status, distance to downtown and to the nearest radial road, a crime rate, and town-level variables reflecting land use (zoning, industry), taxation and education [@bivand17]. This structure will be used here to exercise issues raised in fitting spatial regression models, including the presence of multiple levels.
## Multilevel models of the Boston dataset {#sec-multilevel}
\index{multilevel models}
The ZN, INDUS, NOX, RAD, TAX, and PTRATIO variables show effectively no variability within the TASSIM zones, so in a multilevel model the random effect may absorb their influence.
```{r}
form <- formula(log(median) ~ CRIM + ZN + INDUS + CHAS +
I((NOX*10)^2) + I(RM^2) + AGE + log(DIS) +
log(RAD) + TAX + PTRATIO + I(BB/100) +
log(I(LSTAT/100)))
```
### IID random effects with lme4
\index{lme4}
The **lme4** package [@R-lme4] lets us add an independent and identically distributed (IID) unstructured random effect at the model output zone level by updating the model formula with a random effects term:
```{r, message=FALSE}
library(Matrix)
library(lme4)
MLM <- lmer(update(form, . ~ . + (1 | NOX_ID)),
data = boston_487, REML = FALSE)
```
Copying the random effect into the `"sf"` object for mapping is performed below.
```{r}
boston_93$MLM_re <- ranef(MLM)[[1]][,1]
```
### IID and CAR random effects with hglm
\index{hglm}
The same model may be estimated using the **hglm** package [@R-hglm], which also permits the modelling of discrete responses, this time using an extra one-sided formula to express the random effects term:
```{r}
library(hglm) |> suppressPackageStartupMessages()
suppressWarnings(HGLM_iid <- hglm(fixed = form,
random = ~1 | NOX_ID,
data = boston_487,
family = gaussian()))
boston_93$HGLM_re <- unname(HGLM_iid$ranef)
```
The same package has been extended to spatially structured SAR and CAR random effects, for which a sparse spatial weights matrix is required [@alam-ronnegard-shen:2015]; we choose binary spatial weights:
```{r}
library(spatialreg)
W <- as(spdep::nb2listw(nb_q_93, style = "B"), "CsparseMatrix")
```
We fit a CAR model at the upper level, using the `rand.family` argument, where the values of the indexing variable `NOX_ID` match the row names of $W$:
```{r}
suppressWarnings(HGLM_car <- hglm(fixed = form,
random = ~ 1 | NOX_ID,
data = boston_487,
family = gaussian(),
rand.family = CAR(D=W)))
boston_93$HGLM_ss <- HGLM_car$ranef[,1]
```
### IID and ICAR random effects with R2BayesX
\index{R2BayesX}
The **R2BayesX** package [@R-R2BayesX] provides flexible support for structured additive regression models, including spatial multilevel models. The models include an IID unstructured random effect at the upper level using the `"re"` specification in the `sx` model term [@umlaufetal:15]; we choose the `"MCMC"` method:
```{r}
library(R2BayesX) |> suppressPackageStartupMessages()
```
```{r, cache=TRUE}
BX_iid <- bayesx(update(form, . ~ . + sx(NOX_ID, bs = "re")),
family = "gaussian", data = boston_487,
method = "MCMC", iterations = 12000,
burnin = 2000, step = 2, seed = 123)
```
```{r}
boston_93$BX_re <- BX_iid$effects["sx(NOX_ID):re"][[1]]$Mean
```
and the `"mrf"` (Markov Random Field) spatially structured intrinsic CAR random effect specification based on a graph derived from converting a suitable `"nb"` object for the upper level. The `"region.id"` attribute of the `"nb"` object needs to contain values corresponding to the indexing variable in the `sx` effects term, to facilitate the internal construction of design matrix $Z$:
```{r}
RBX_gra <- nb2gra(nb_q_93)
all.equal(row.names(RBX_gra), attr(nb_q_93, "region.id"))
```
As we saw above in the intrinsic CAR model definition, the counts of neighbours are entered on the diagonal, but the current implementation uses a dense, not sparse, matrix:
```{r}
all.equal(unname(diag(RBX_gra)), spdep::card(nb_q_93))
```
The `sx` model term continues to include the indexing variable, and now passes through the intrinsic CAR precision matrix:
```{r, cache=TRUE}
BX_mrf <- bayesx(update(form, . ~ . + sx(NOX_ID, bs = "mrf",
map = RBX_gra)),
family = "gaussian", data = boston_487,
method = "MCMC", iterations = 12000,
burnin = 2000, step = 2, seed = 123)
```
```{r}
boston_93$BX_ss <- BX_mrf$effects["sx(NOX_ID):mrf"][[1]]$Mean
```
### IID, ICAR and Leroux random effects with INLA
\index{INLA}
@JSSv063i20 and @gómez2020bayesian present the use of the **INLA** package [@R-INLA] and the `inla` model fitting function with spatial regression models:
```{r}
library(INLA) |> suppressPackageStartupMessages()
```
Although differing in details, the approach by updating the fixed model formula with an unstructured random effects term is very similar to that seen above:
```{r, cache=TRUE, eval = eval_inla}
INLA_iid <- inla(update(form, . ~ . + f(NOX_ID, model = "iid")),
family = "gaussian", data = boston_487)
```
```{r, eval = eval_inla}
boston_93$INLA_re <- INLA_iid$summary.random$NOX_ID$mean
```
As with most implementations, care is needed to match the indexing variable with the spatial weights; in this case using indices $1, \dots, 93$ rather than the `NOX_ID` variable directly:
```{r}
ID2 <- as.integer(as.factor(boston_487$NOX_ID))
```
The same sparse binary spatial weights matrix is used, and the intrinsic CAR representation is constructed internally:
```{r, cache=TRUE,warning=FALSE, eval = eval_inla}
INLA_ss <- inla(update(form, . ~ . + f(ID2, model = "besag",
graph = W)),
family = "gaussian", data = boston_487)
```
```{r, eval = eval_inla}
boston_93$INLA_ss <- INLA_ss$summary.random$ID2$mean
```
The sparse Leroux representation as given by @gómez2020bayesian can be constructed in the following way:
```{r}
M <- Diagonal(nrow(W), rowSums(W)) - W
Cmatrix <- Diagonal(nrow(M), 1) - M
```
This model can be estimated using the `"generic1"` model with the specified precision matrix:
```{r, cache=TRUE, warning=FALSE, eval = eval_inla}
INLA_lr <- inla(update(form, . ~ . + f(ID2, model = "generic1",
Cmatrix = Cmatrix)),
family = "gaussian", data = boston_487)
```
```{r, eval = eval_inla}
boston_93$INLA_lr <- INLA_lr$summary.random$ID2$mean
```
### ICAR random effects with mgcv::gam()
\index{mgcv gam!ICAR}
In a very similar way, the `gam` function in the **mgcv** package [@R-mgcv] can take an `"mrf"` term using a suitable `"nb"` object for the upper level. In this case the `"nb"` object needs to have the contents of the `"region.id"` attribute copied as the names of the neighbour list components, and the indexing variable needs to be a factor [@wood:17]:
```{r}
library(mgcv)
names(nb_q_93) <- attr(nb_q_93, "region.id")
boston_487$NOX_ID <- as.factor(boston_487$NOX_ID)
```
The specification of the spatially structured term again differs in details from those above, but achieves the same purpose. The `"REML"` method of `bayesx` gives the same results as `gam` using `"REML"` in this case:
```{r, cache=TRUE}
GAM_MRF <- gam(update(form, . ~ . + s(NOX_ID, bs = "mrf",
xt = list(nb = nb_q_93))),
data = boston_487, method = "REML")
```
The upper-level random effects may be extracted by predicting terms; as we can see, the values in all lower-level tracts belonging to the same upper-level air pollution model output zones are identical:
```{r}
ssre <- predict(GAM_MRF, type = "terms",
se = FALSE)[, "s(NOX_ID)"]
all(sapply(tapply(ssre, list(boston_487$NOX_ID), c),
function(x) length(unique(round(x, 8))) == 1))
```
so we can return the first value for each upper-level unit:
```{r}
boston_93$GAM_ss <- aggregate(ssre, list(boston_487$NOX_ID),
head, n=1)$x
```
### Upper-level random effects: summary
In the cases of `hglm`, `bayesx`, `inla` and `gam`, we could also model discrete responses without further major difficulty, and `bayesx`, `inla` and `gam` also facilitate the generalisation of functional form fitting for included covariates.
Unfortunately, the coefficient estimates for the air pollution variable for these multilevel models are not helpful. All are negative as expected, but the inclusion of the model output zone level effects, IID or spatially structured, makes it is hard to disentangle the influence of the scale of observation from that of covariates observed at that scale rather than at the tract level.
@fig-multi-levelmaps1 shows that the air pollution model output zone level IID random effects are very similar across the four model fitting functions reported. In all the maps, the central downtown zones have stronger negative random effect values, but strong positive values are also found in close proximity; suburban areas take values closer to zero.
```{r fig-multi-levelmaps1, echo=!knitr::is_latex_output(), message=FALSE, eval = eval_inla}
#| code-fold: true
#| fig.cap: "Air pollution model output zone level IID random effects estimated using **lme4**, **hglm**, **INLA** and **R2BayesX**; the range of the response, `log(median)` is 2.1893"
library(tmap, warn.conflicts=FALSE)
tmap4 <- packageVersion("tmap") >= "3.99"
if (tmap4) {
tm_shape(boston_93) +
tm_polygons(fill = c("MLM_re", "HGLM_re", "INLA_re", "BX_re"),
fill.legend = tm_legend("IID", frame=FALSE, item.r = 0),
fill.free = FALSE, lwd = 0.01,
fill.scale = tm_scale(midpoint = 0, values = "brewer.rd_yl_gn")) +
tm_facets_wrap(columns = 2, rows = 2) +
tm_layout(panel.labels = c("lmer", "hglm", "inla", "bayesx"))
} else {
tm_shape(boston_93) +
tm_fill(c("MLM_re", "HGLM_re", "INLA_re", "BX_re"),
midpoint = 0, title = "IID") +
tm_facets(free.scales = FALSE) +
tm_borders(lwd = 0.3, alpha = 0.4) +
tm_layout(panel.labels = c("lmer", "hglm", "inla", "bayesx"))
}
```
@fig-multi-levelmaps2 shows that the spatially structured random effects are also very similar to each other, with the `"SAR"` spatial smooth being perhaps a little smoother than the `"CAR"` smooths when considering the range of values taken by the random effect term.
```{r fig-multi-levelmaps2, echo=!knitr::is_latex_output(), message=FALSE, eval = eval_inla}
#| code-fold: true
#| fig.cap: "Air pollution model output zone level spatially structured random effects estimated using **hglm**, **HSAR**, **INLA**, **R2BayesX** and **mgcv**"
if (tmap4) {
tm_shape(boston_93) +
tm_polygons(fill = c("HGLM_ss", "INLA_lr", "INLA_ss", "BX_ss",
"GAM_ss"),
fill.legend = tm_legend("SSRE", frame=FALSE, item.r = 0),
fill.free = FALSE, lwd = 0.1,
fill.scale = tm_scale(midpoint = 0, values = "brewer.rd_yl_gn")) +
tm_facets_wrap(columns = 3, rows = 2) +
tm_layout(panel.labels = c("hglm CAR", "inla Leroux",
"inla ICAR", "bayesx ICAR", "gam ICAR"))
} else {
tm_shape(boston_93) +
tm_fill(c("HGLM_ss", "INLA_lr", "INLA_ss", "BX_ss",
"GAM_ss"), midpoint = 0, title = "SSRE") +
tm_facets(free.scales = FALSE) +
tm_borders(lwd = 0.3, alpha = 0.4) +
tm_layout(panel.labels = c("hglm CAR", "inla Leroux",
"inla ICAR", "bayesx ICAR", "gam ICAR"))
}
```
Although there is still a great need for more thorough comparative studies of model fitting functions for spatial regression including multilevel capabilities, there has been much progress over recent years. @VRANCKX2019100302 offer a recent comparative survey of disease mapping spatial regression, typically set in a Poisson regression framework offset by an expected count. In @doi:10.1177/1471082X20967158, methods for estimating spatial survival models using spatial weights matrices are compared with spatial probit models.
## Exercises
1. Construct a multilevel dataset using the Athens housing data from the archived **HSAR** package: <https://cran.r-project.org/src/contrib/Archive/HSAR/HSAR_0.5.1.tar.gz>, and included in **spData** from version 2.2.1. At which point do the municipality department attribute values get copied out to all the point observations within each municipality department?
2. Create neighbour objects at both levels. Test `greensp` for spatial autocorrelation at the upper level, and then at the lower level. What has been the chief consequence of copying out the area of green spaces in square meters for the municipality departments to the point support property level?
3. Using the formula object from the vignette, assess whether adding the copied out upper-level variables seems sensible. Use `mgcv::gam` to fit a linear mixed effects model (IID of `num_dep` identifying the municipality departments) using just the lower-level variables and the lower- and upper-level variables. Do your conclusions differ?
4. Complete the analysis by replacing the IID random effects with an `"mrf"` Markov random field and the contiguity neighbour object created above. Do you think that it is reasonable to, for example, draw any conclusions based on the municipality department level variables such as `greensp`?
```{r echo=FALSE}
save(list = ls(), file = "ch16.RData")
```