This repository has been archived by the owner on Jun 27, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathglmm.qmd
378 lines (299 loc) · 14.2 KB
/
glmm.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
---
title: "Generalized linear mixed models"
jupyter: julia-1.9
---
Load the packages to be used
```{julia}
#| code-fold: true
#| output: false
using AlgebraOfGraphics
using CairoMakie
using DataFrameMacros
using DataFrames
using MixedModels
using MixedModelsMakie
using SMLP2023: dataset
CairoMakie.activate!(; type="svg")
import ProgressMeter
ProgressMeter.ijulia_behavior(:clear)
```
- A GLMM (Generalized Linear Mixed Model) is used instead of a LMM (Linear Mixed Model) when the response is binary or, perhaps, a count with a low expected count.
- The specification of the model includes the distribution family for the response and, possibly, the *link function*, g, relating the *mean response*, μ, to the value of the *linear predictor*, η.
- To explain the model it helps to consider the linear mixed model in some detail first.
## Matrix notation for the sleepstudy model
```{julia}
sleepstudy = DataFrame(dataset(:sleepstudy))
```
```{julia}
contrasts = Dict(:subj => Grouping())
m1 = let f = @formula reaction ~ 1 + days + (1 + days | subj)
fit(MixedModel, f, sleepstudy; contrasts)
end
println(m1)
```
The response vector, y, has 180 elements. The fixed-effects coefficient vector, β, has 2 elements and the fixed-effects model matrix, X, is of size 180 × 2.
```{julia}
m1.y
```
```{julia}
m1.β
```
```{julia}
m1.X
```
The second column of X is just the `days` vector and the first column is all 1's.
There are 36 random effects, 2 for each of the 18 levels of `subj`.
The "estimates" (technically, the conditional means or conditional modes) are returned as a vector of matrices, one matrix for each grouping factor.
In this case there is only one grouping factor for the random effects so there is one one matrix which contains 18 intercept random effects and 18 slope random effects.
```{julia}
m1.b
```
```{julia}
only(m1.b) # only one grouping factor
```
There is a model matrix, Z, for the random effects.
In general it has one chunk of columns for the first grouping factor, a chunk of columns for the second grouping factor, etc.
In this case there is only one grouping factor.
```{julia}
Int.(first(m1.reterms))
```
The defining property of a linear model or linear mixed model is that the fitted values are linear combinations of the fixed-effects parameters and the random effects.
We can write the fitted values as
```{julia}
m1.X * m1.β + only(m1.reterms) * vec(only(m1.b))
```
```{julia}
fitted(m1) # just to check that these are indeed the same as calculated above
```
In symbols we would write the *linear predictor expression* as
$$
\boldsymbol{\eta} = \mathbf{X}\boldsymbol{\beta} +\mathbf{Z b}
$$
where $\boldsymbol{\eta}$ has 180 elements, $\boldsymbol{\beta}$ has 2 elements, $\bf b$ has 36 elements, $\bf X$ is of size 180 × 2 and $\bf Z$ is of size 180 × 36.
For a linear model or linear mixed model the linear predictor **is** the mean response, $\boldsymbol\mu$.
That is, we can write the probability model in terms of a 180-dimensional random variable, $\mathcal Y$, for the response and a 36-dimensional random variable, $\mathcal B$, for the random effects as
$$
\begin{aligned}
(\mathcal{Y} | \mathcal{B}=\bf{b}) &\sim\mathcal{N}(\bf{ X\boldsymbol\beta + Z b},\sigma^2\bf{I})\\\\
\mathcal{B}&\sim\mathcal{N}(\bf{0},\boldsymbol{\Sigma}_{\boldsymbol\theta}) .
\end{aligned}
$$
where $\boldsymbol{\Sigma}_\boldsymbol{\theta}$ is a 36 × 36 symmetric covariance matrix that has a special form - it consists of 18 diagonal blocks, each of size 2 × 2 and all the same.
Recall that this symmetric matrix can be constructed from the parameters $\boldsymbol\theta$, which generate the lower triangular matrix $\boldsymbol\lambda$, and the estimate $\widehat{\sigma^2}$.
```{julia}
m1.θ
```
```{julia}
λ = only(m1.λ) # with multiple grouping factors there will be multiple λ's
```
```{julia}
Σ = varest(m1) * (λ * λ')
```
Compare the diagonal elements to the *Variance* column of
```{julia}
VarCorr(m1)
```
## Linear predictors in LMMs and GLMMs
Writing the model for $\mathcal Y$ as
$$
(\mathcal{Y} | \mathcal{B}=\bf{b})\sim\mathcal{N}(\bf{ X\boldsymbol\beta + Z b},\sigma^2\bf{I})
$$
may seem like over-mathematization (or "overkill", if you prefer) relative to expressions like
$$
y_i = \beta_1 x_{i,1} + \beta_2 x_{i,2}+ b_1 z_{i,1} +\dots+b_{36} z_{i,36}+\epsilon_i
$$
but this more abstract form is necessary for generalizations.
The way that I read the first form is
:::{.callout}
The conditional distribution of the response vector, $\mathcal Y$, given that the random effects vector, $\mathcal B =\bf b$, is a multivariate normal (or *Gaussian*) distribution whose mean, $\boldsymbol\mu$, is the linear predictor, $\boldsymbol\eta=\bf{X\boldsymbol\beta+Zb}$, and whose covariance matrix is $\sigma^2\bf I$.
That is, conditional on $\bf b$, the elements of $\mathcal Y$ are independent normal random variables with constant variance, $\sigma^2$, and means of the form $\boldsymbol\mu = \boldsymbol\eta = \bf{X\boldsymbol\beta+Zb}$.
:::
So the only things that differ in the distributions of the $y_i$'s are the means and they are determined by this linear predictor, $\boldsymbol\eta = \bf{X\boldsymbol\beta+Zb}$.
## Generalized Linear Mixed Models
Consider first a GLMM for a vector, $\bf y$, of binary (i.e. yes/no) responses.
The probability model for the conditional distribution $\mathcal Y|\mathcal B=\bf b$ consists of independent [Bernoulli distributions](https://en.wikipedia.org/wiki/Bernoulli_distribution) where the mean, $\mu_i$, for the i'th response is again determined by the i'th element of a *linear predictor*, $\boldsymbol\eta = \mathbf{X}\boldsymbol\beta+\mathbf{Z b}$.
However, in this case we will run into trouble if we try to make $\boldsymbol\mu=\boldsymbol\eta$ because $\mu_i$ is the probability of "success" for the i'th response and must be between 0 and 1.
We can't guarantee that the i'th component of $\boldsymbol\eta$ will be between 0 and 1.
To get around this problem we apply a transformation to take $\eta_i$ to $\mu_i$.
For historical reasons this transformation is called the *inverse link*, written $g^{-1}$, and the opposite transformation - from the probability scale to an unbounded scale - is called the *link*, g.
Each probability distribution in the [exponential family](https://en.wikipedia.org/wiki/Exponential_family) (which is most of the important ones), has a *canonical link* which comes from the form of the distribution itself. The details aren't as important as recognizing that the distribution itself determines a preferred link function.
For the Bernoulli distribution, the canonical link is the *logit* or *log-odds* function,
$$
\eta = g(\mu) = \log\left(\frac{\mu}{1-\mu}\right),
$$
(it's called *log-odds* because it is the logarithm of the odds ratio, $p/(1-p)$)
and the canonical inverse link is the *logistic*
$$
\mu=g^{-1}(\eta)=\frac{1}{1+\exp(-\eta)}.
$$
This is why fitting a binary response is sometimes called *logistic regression*.
For later use we define a Julia `logistic` function.
See [this presentation](https://github.com/dmbates/JSM2021/blob/main/notebooks/2compilation.jl) for more information than you could possibly want to know on how Julia converts code like this to run on the processor.
```{julia}
increment(x) = x + one(x)
logistic(η) = inv(increment(exp(-η)))
```
To reiterate, the probability model for a Generalized Linear Mixed Model (GLMM) is
$$
\begin{aligned}
(\mathcal{Y} | \mathcal{B}=\bf{b}) &\sim\mathcal{D}(\bf{g^{-1}(X\boldsymbol\beta + Z b)},\phi)\\\\
\mathcal{B}&\sim\mathcal{N}(\bf{0},\Sigma_{\boldsymbol\theta}) .
\end{aligned}
$$
where $\mathcal{D}$ is the distribution family (such as Bernoulli or Poisson), $g^{-1}$ is the inverse link and $\phi$ is a scale parameter for $\mathcal{D}$ if it has one.
The important cases of the Bernoulli and Poisson distributions don't have a scale parameter - once you know the mean you know everything you need to know about the distribution. (For those following the presentation, [this poem](https://www.poetryfoundation.org/poems/44477/ode-on-a-grecian-urn) by John Keats is the one with the couplet "Beauty is truth, truth beauty - that is all ye know on earth and all ye need to know.")
### An example of a Bernoulli GLMM
The `contra` dataset in the `MixedModels` package is from a survey on the use of artificial contraception by women in Bangladesh.
```{julia}
contra = DataFrame(dataset(:contra))
```
```{julia}
combine(groupby(contra, :dist), nrow)
```
The information recorded included woman's age, the number of live children she has, whether she lives in an urban or rural setting, and the political district in which she lives.
The age was centered.
Unfortunately, the version of the data to which I had access did not record what the centering value was.
A data plot, @fig-contradata, shows that the probability of contraception use is **not** linear in `age` - it is low for younger women, higher for women in the middle of the range (assumed to be women in late 20's to early 30's) and low again for older women (late 30's to early 40's in this survey).
If we fit a model with only the `age` term in the fixed effects, that term will not be significant.
This doesn't mean that there is no "age effect", it only means that there is no significant linear effect for `age`.
```{julia}
#| code-fold: true
#| fig-cap: Smoothed relative frequency of contraception use versus centered age for women in the 1989 Bangladesh Fertility Survey
#| label: fig-contradata
draw(
data(
@transform(
contra,
:numuse = Int(:use == "Y"),
:urb = ifelse(:urban == "Y", "Urban", "Rural")
)
) *
mapping(
:age => "Centered age (yr)",
:numuse => "Frequency of contraception use";
col=:urb,
color=:livch,
) *
smooth();
figure=(; resolution=(800, 450)),
)
```
```{julia}
contrasts = Dict(
:dist => Grouping(),
:urban => HelmertCoding(),
:livch => DummyCoding(), # default, but no harm in being explicit
)
nAGQ = 9
dist = Bernoulli()
gm1 = let
form = @formula(
use ~ 1 + age + abs2(age) + urban + livch + (1 | dist)
)
fit(MixedModel, form, contra, dist; nAGQ, contrasts)
end
```
Notice that the linear term for `age` is not significant but the quadratic term for `age` is highly significant.
We usually retain the lower order term, even if it is not significant, if the higher order term is significant.
Notice also that the parameter estimates for the treatment contrasts for `livch` are similar.
Thus the distinction of 1, 2, or 3+ children is not as important as the contrast between having any children and not having any.
Those women who already have children are more likely to use artificial contraception.
Furthermore, the women without children have a different probability vs age profile than the women with children.
To allow for this we define a binary `children` factor and incorporate an `age&children` interaction.
```{julia}
VarCorr(gm1)
```
Notice that there is no "residual" variance being estimated.
This is because the Bernoulli distribution doesn't have a scale parameter.
### Convert `livch` to a binary factor
```{julia}
@transform!(contra, :children = :livch ≠ "0")
# add the associated contrast specifier
contrasts[:children] = EffectsCoding()
```
```{julia}
gm2 = let
form = @formula(
use ~
1 +
age * children +
abs2(age) +
children +
urban +
(1 | dist)
)
fit(MixedModel, form, contra, dist; nAGQ, contrasts)
end
```
```{julia}
#| code-fold: true
let
mods = [gm2, gm1]
DataFrame(;
model=[:gm2, :gm1],
npar=dof.(mods),
deviance=deviance.(mods),
AIC=aic.(mods),
BIC=bic.(mods),
AICc=aicc.(mods),
)
end
```
Because these models are not nested, we cannot do a likelihood ratio test.
Nevertheless we see that the deviance is much lower in the model with `age & children` even though the 3 levels of `livch` have been collapsed into a single level of `children`.
There is a substantial decrease in the deviance even though there are fewer parameters in model `gm2` than in `gm1`.
This decrease is because the flexibility of the model - its ability to model the behavior of the response - is being put to better use in `gm2` than in `gm1`.
At present the calculation of the `geomdof` as `sum(influence(m))` is not correctly defined in our code for a GLMM so we need to do some more work before we can examine those values.
### Using `urban&dist` as a grouping factor
It turns out that there can be more difference between urban and rural settings within the same political district than there is between districts.
To model this difference we build a model with `urban&dist` as a grouping factor.
```{julia}
gm3 = let
form = @formula(
use ~
1 +
age * children +
abs2(age) +
children +
urban +
(1 | urban & dist)
)
fit(MixedModel, form, contra, dist; nAGQ, contrasts)
end
```
```{julia}
#| code-fold: true
let
mods = [gm3, gm2, gm1]
DataFrame(;
model=[:gm3, :gm2, :gm1],
npar=dof.(mods),
deviance=deviance.(mods),
AIC=aic.(mods),
BIC=bic.(mods),
AICc=aicc.(mods),
)
end
```
Notice that the parameter count in `gm3` is the same as that of `gm2` - the thing that has changed is the number of levels of the grouping factor- resulting in a much lower deviance for `gm3`.
This reinforces the idea that a simple count of the number of parameters to be estimated does not always reflect the complexity of the model.
```{julia}
gm2
```
```{julia}
gm3
```
The coefficient for `age` may be regarded as insignificant but we retain it for two reasons: we have a term of `age²` (written `abs2(age)`) in the model and we have a significant interaction `age & children` in the model.
### Predictions for some subgroups
For a "typical" district (random effect near zero) the predictions on the linear predictor scale for a woman whose age is near the centering value (i.e. centered age of zero) are:
```{julia}
using Effects
design = Dict(
:children => [true, false], :urban => ["Y", "N"], :age => [0.0]
)
preds = effects(design, gm3; invlink=AutoInvLink())
```
## Summarizing the results
- From the data plot we can see a quadratic trend in the probability by age.
- The patterns for women with children are similar and we do not need to distinguish between 1, 2, and 3+ children.
- We do distinguish between those women who do not have children and those with children. This shows up in a significant `age & children` interaction term.