This vignette gives a few examples of the use of the emmeans package to analyze other than the basic types of models provided by the stats package. Emphasis here is placed on accessing the optional capabilities that are typically not needed for the more basic models. A reference for all supported models is provided in the “models” vignette.
Linear mixed models are really important in statistics. Emphasis here is placed on those fitted using lme4::lmer(), but emmeans also supports other mixed-model packages such as nlme.
To illustrate, consider the Oats dataset in the nlme package. It has the results of a balanced split-plot experiment: experimental blocks are divided into plots that are randomly assigned to oat varieties, and the plots are subdivided into subplots that are randomly assigned to amounts of nitrogen within each plot. We will consider a linear mixed model for these data, excluding interaction (which is justified in this case). For sake of illustration, we will exclude a few observations.
Oats.lmer <- lme4::lmer(yield ~ Variety + factor(nitro) + (1|Block/Variety),
                        data = nlme::Oats, subset = -c(1,2,3,5,8,13,21,34,55))Let’s look at the EMMs for nitro:
Oats.emmGrid.n <- emmeans(Oats.lmer, "nitro")
Oats.emmGrid.n
##  nitro    emmean       SE   df  lower.CL  upper.CL
##    0.0  78.89207 7.294379 7.78  61.98930  95.79484
##    0.2  97.03425 7.136271 7.19  80.49785 113.57065
##    0.4 114.19816 7.136186 7.19  97.66196 130.73437
##    0.6 124.06857 7.070235 6.95 107.68519 140.45195
## 
## Results are averaged over the levels of: Variety 
## Degrees-of-freedom method: kenward-roger 
## Confidence level used: 0.95You will notice that the degrees of freedom are fractional: that is due to the fact that whole-plot and subplot variations are combined when standard errors are estimated. Different degrees-of-freedom methods are available. By default, the Kenward-Roger method is used, and that’s why you see a message about the pbkrtest package being loaded, as it implements that method. We may specify a different degrees-of-freedom method via the optional argument lmer.df:
emmeans(Oats.lmer, "nitro", lmer.df = "satterthwaite")
## Loading required namespace: lmerTest
##  nitro    emmean       SE   df  lower.CL  upper.CL
##    0.0  78.89207 7.280608 7.28  61.81122  95.97292
##    0.2  97.03425 7.128706 6.72  80.30977 113.75873
##    0.4 114.19816 7.128885 6.72  97.47326 130.92306
##    0.6 124.06857 7.067304 6.49 107.48815 140.64899
## 
## Results are averaged over the levels of: Variety 
## Degrees-of-freedom method: satterthwaite 
## Confidence level used: 0.95This latest result uses the Satterthwaite method, which is implemented in the lmerTest package. Note that, with this method, not only are the degrees of freedom slightly different, but so are the standard errors. That is because the Kenward-Roger method also entails making a bias adjustment to the covariance matrix of the fixed effects; that is the principal difference between the methods. A third possibility is "asymptotic":
emmeans(Oats.lmer, "nitro", lmer.df = "asymptotic")
##  nitro    emmean       SE df asymp.LCL asymp.UCL
##    0.0  78.89207 7.280608 NA  64.62234   93.1618
##    0.2  97.03425 7.128706 NA  83.06225  111.0063
##    0.4 114.19816 7.128885 NA 100.22580  128.1705
##    0.6 124.06857 7.067304 NA 110.21691  137.9202
## 
## Results are averaged over the levels of: Variety 
## Degrees-of-freedom method: asymptotic 
## Confidence level used: 0.95This just sets all the degrees of freedom to NA – that’s emmeans’s way of using z statistics rather than t statistics. The asymptotic methods tend to make confidence intervals a bit too narrow and P values a bit too low; but they involve much, much less computation. Note that the SEs are the same as obtained using the Satterthwaite method.
Comparisons and contrasts are pretty much the same as with other models. As nitro has quantitative levels, we might want to test polynomial contrasts:
contrast(Oats.emmGrid.n, "poly")
##  contrast    estimate        SE    df t.ratio p.value
##  linear    152.693407 15.577738 43.23   9.802  <.0001
##  quadratic  -8.271775  6.948552 44.19  -1.190  0.2402
##  cubic      -6.315230 15.205089 42.75  -0.415  0.6800
## 
## Results are averaged over the levels of: VarietyThe interesting thing here is that the degrees of freedom are much larger than they are for the EMMs. The reason is because nitro within-plot factor, so inter-plot variations have little role in estimating contrasts among nitro levels. On the other hand, Variety is a whole-plot factor, and there is not much of a bump in degrees of freedom for comparisons:
emmeans(Oats.lmer, pairwise ~ Variety)
## $emmeans
##  Variety        emmean       SE   df lower.CL upper.CL
##  Golden Rain 105.24081 7.531718 8.46 88.03704 122.4446
##  Marvellous  108.46951 7.482632 8.28 91.37786 125.5612
##  Victory      96.93446 7.641645 8.81 79.47960 114.3893
## 
## Results are averaged over the levels of: nitro 
## Degrees-of-freedom method: kenward-roger 
## Confidence level used: 0.95 
## 
## $contrasts
##  contrast                  estimate       SE   df t.ratio p.value
##  Golden Rain - Marvellous -3.228698 6.553848 9.56  -0.493  0.8764
##  Golden Rain - Victory     8.306351 6.707935 9.80   1.238  0.4595
##  Marvellous - Victory     11.535049 6.670488 9.80   1.729  0.2431
## 
## Results are averaged over the levels of: nitro 
## P value adjustment: tukey method for comparing a family of 3 estimatesThe computation required to compute the adjusted covariance matrix and degrees of freedom may become cumbersome. Some user options (i.e., emm_options() calls) make it possible to streamline these computations through default methods and limitations on them. First, the option lmer.df, which may have values of "kenward-roger", "satterthwaite", or "asymptotic" (partial matches are OK!) specifies the default degrees-of-freedom method.
The options disable.pbkrtest and disable.lmerTest may be TRUE or FALSE, and comprise another way of controlling which method is used (e.g., the Kenward-Roger method will not be used if get_emm_option("disable.pbkrtest") == TRUE). Finally, the options pbkrtest.limit and lmerTest.limit, which should be set to numeric values, enable the given package conditionally on whether the number of data rows does not exceed the given limit. The factory default is 3000 for both limits.
Ordinal-response models comprise an example where several options are available for obtaining EMMs. To illustrate, consider the wine data in the ordinal package. The response is a rating of bitterness on a five-point scale. we will consider a probit model in two factors during fermentation: temp (temperature) and contact (contact with grape skins), with the judge making the rating as a scale predictor:
require("ordinal")
## Loading required package: ordinal
wine.clm <- clm(rating ~ temp + contact, scale = ~ judge,
                data = wine, link = "probit")(in earlier modeling, we found little interaction between the factors.) Here are the EMMs for each factor using default options:
emmeans(wine.clm, list(pairwise ~ temp, pairwise ~ contact))
## $`emmeans of temp`
##  temp     emmean        SE df  asymp.LCL  asymp.UCL
##  cold -0.8838870 0.2898157 NA -1.4519155 -0.3158586
##  warm  0.6011128 0.2246028 NA  0.1608993  1.0413263
## 
## Results are averaged over the levels of: contact, judge 
## Confidence level used: 0.95 
## 
## $`pairwise differences of temp`
##  contrast     estimate        SE df z.ratio p.value
##  cold - warm -1.073545 0.4215306 NA  -2.547  0.0109
## 
## Results are averaged over the levels of: contact, judge 
## 
## $`emmeans of contact`
##  contact     emmean        SE df   asymp.LCL   asymp.UCL
##  no      -0.6143569 0.2983133 NA -1.19904027 -0.02967357
##  yes      0.3315827 0.2014363 NA -0.06322522  0.72639062
## 
## Results are averaged over the levels of: temp, judge 
## Confidence level used: 0.95 
## 
## $`pairwise differences of contact`
##  contrast   estimate        SE df z.ratio p.value
##  no - yes -0.6838446 0.3038508 NA  -2.251  0.0244
## 
## Results are averaged over the levels of: temp, judgeThese results are on the “latent” scale; the idea is that there is a continuous random variable (in this case normal, due to the probit link) having a mean that depends on the predictors; and that the ratings are a discretization of the latent variable based on a fixed set of cut points (which are estimated). In this particular example, we also have a scale model that says that the variance of the latent variable depends on the judges. The latent results are quite a bit like those for measurement data, making them easy to interpret. The only catch is that they are not uniquely defined: we could apply a linear transformation to them, and the same linear transformation to the cut points, and the results would be the same.
The clm function actually fits the model using an ordinary probit model but with different intercepts for each cut point. We can get detailed information for this model by specifying mode = "linear.predictor":
tmp <- ref_grid(wine.clm, mode = "lin")
tmp
## 'emmGrid' object with variables:
##     temp = cold, warm
##     contact = no, yes
##     judge = 1, 2, 3, 4, 5, 6, 7, 8, 9
##     cut = multivariate response levels: 1|2, 2|3, 3|4, 4|5
## Transformation: "probit"Note that this reference grid involves an additional constructed predictor named cut that accounts for the different intercepts in the model. Let’s obtain EMMs for temp on the linear-predictor scale:
emmeans(tmp, "temp")
##  temp     emmean        SE df  asymp.LCL  asymp.UCL
##  cold  0.8838870 0.2898157 NA  0.3158586  1.4519155
##  warm -0.6011128 0.2246028 NA -1.0413263 -0.1608993
## 
## Results are averaged over the levels of: contact, judge, cut 
## Results are given on the probit (not the response) scale. 
## Confidence level used: 0.95These are just the negatives of the latent results obtained earlier (the sign is changed to make the comparisons go the right direction). Closely related to this is mode = "cum.prob" and mode = "exc.prob", which simply transform the linear predictor to cumulative probabilities and exceedance (1 - cumulative) probabilities. These modes give us access to the details of the fitted model but are cumbersome to use for describing results. When they can become useful is when you want to work in terms of a particular cut point. Let’s look at temp again in terms of the probability that the rating will be at least 4:
emmeans(wine.clm, ~ temp, mode = "exc.prob", at = list(cut = "3|4"))
##  temp   exc.prob         SE df  asymp.LCL asymp.UCL
##  cold 0.07477754 0.03183196 NA 0.01238804 0.1371670
##  warm 0.40688371 0.07056873 NA 0.26857154 0.5451959
## 
## Results are averaged over the levels of: contact, judge 
## Confidence level used: 0.95There are yet more modes! With mode = "prob", we obtain estimates of the probability distribution of each rating. Its reference grid includes a factor with the same name as the model response – in this case rating. We usually want to use that as the primary factor, and the factors of interest as by variables:
emmeans(wine.clm, ~ rating | temp, mode = "prob")
## temp = cold:
##  rating       prob         SE df    asymp.LCL  asymp.UCL
##  1      0.12922075 0.06252864 NA  0.006666872 0.25177462
##  2      0.48768408 0.07051118 NA  0.349484709 0.62588345
##  3      0.30831763 0.05942035 NA  0.191855896 0.42477937
##  4      0.05766120 0.02378461 NA  0.011044211 0.10427818
##  5      0.01711634 0.01265193 NA -0.007680992 0.04191368
## 
## temp = warm:
##  rating       prob         SE df    asymp.LCL  asymp.UCL
##  1      0.01561393 0.01287141 NA -0.009613574 0.04084144
##  2      0.14730458 0.04475283 NA  0.059590654 0.23501851
##  3      0.43019777 0.06274049 NA  0.307228674 0.55316687
##  4      0.26845558 0.06251632 NA  0.145925842 0.39098532
##  5      0.13842813 0.05061119 NA  0.039232030 0.23762424
## 
## Results are averaged over the levels of: contact, judge 
## Confidence level used: 0.95Using mode = "mean.class" obtains the average of these probability distributions as probabilities of the integers 1–5:
emmeans(wine.clm, "temp", mode = "mean.class")
##  temp mean.class        SE df asymp.LCL asymp.UCL
##  cold   2.345768 0.1438341 NA  2.063859  2.627678
##  warm   3.366779 0.1462579 NA  3.080119  3.653440
## 
## Results are averaged over the levels of: contact, judge 
## Confidence level used: 0.95And there is a mode for the scale model too. In this example, the scale model involves only judges, and that is the only factor in the grid:
summary(ref_grid(wine.clm, mode = "scale"), type = "response")
##  judge  response        SE df
##  1     1.0000000 0.0000000 NA
##  2     1.0429804 0.5696650 NA
##  3     1.0526798 0.4811605 NA
##  4     0.7103256 0.3356562 NA
##  5     0.6632397 0.3009780 NA
##  6     0.7582403 0.3411929 NA
##  7     1.0707070 0.5861427 NA
##  8     0.2409019 0.1791135 NA
##  9     0.5331235 0.3110024 NAJudge 8’s ratings don’t vary much, relative to the others. The scale model is in terms of log(SD). Again, these are not uniquely identifiable, and the first level’s estimate is set to log(1) = 0. so, actually, each estimate shown is a comparison with judge 1.
To illustrate emmeans’s support for models fitted using MCMC methods, consider the example_model available in the rstanarm package. The example concerns CBPP, a serious disease of cattle in Ethiopia. A generalized linear mixed model was fitted to the data using the code below. We subsequently obtain its reference grid in the usual way. 
example_model <- rstanarm::stan_glmer(
    cbind(incidence, size - incidence) ~ size + period + (1|herd),
    data = lme4::cbpp, family = binomial,
    chains = 2, cores = 1, seed = 12345, iter = 500)
rst_ex.rg <- ref_grid(example_model)Here is the structure of the reference grid:
rst_ex.rg
## 'emmGrid' object with variables:
##     size = 15.036
##     period = 1, 2, 3, 4
## Transformation: "logit"And, again in the usual way, we can obtain EMMs:
period.emmGrid <- emmeans(rst_ex.rg, "period")
period.emmGrid
## This is a frequentist summary. See `?as.mcmc.emmGrid' for more on what you can do.
##  period    emmean        SE df asymp.LCL asymp.UCL
##  1      -1.450956 0.2825317 NA -2.004708 -0.897204
##  2      -2.405356 0.3373780 NA -3.066605 -1.744108
##  3      -2.514019 0.3408710 NA -3.182114 -1.845924
##  4      -2.985176 0.4633889 NA -3.893402 -2.076951
## 
## Results are given on the logit (not the response) scale. 
## Confidence level used: 0.95The summary shows a frequentist summary of the EMMs. Under the hood, though, is a posterior sample of 500 sets of EMMs – each one being the EMMs computed from each set of parameter values in the posterior sample of regression coefficients. The summary we see is based on the sample means and covariances of that sample.
We can access these posterior results via the as.mcmc method for emmGrid objects. This gives us an object of class mcmc (defined in the coda package), which can be summarized and explored as we please.
require("coda")  ### needed to access generic for as.mcmc()
## Loading required package: coda
summary(as.mcmc(period.emmGrid))
## 
## Iterations = 1:250
## Thinning interval = 1 
## Number of chains = 2 
## Sample size per chain = 250 
## 
## 1. Empirical mean and standard deviation for each variable,
##    plus standard error of the mean:
## 
##            Mean     SD Naive SE Time-series SE
## period 1 -1.435 0.2825  0.01264        0.01771
## period 2 -2.393 0.3374  0.01509        0.01620
## period 3 -2.516 0.3409  0.01524        0.01540
## period 4 -2.979 0.4634  0.02072        0.02793
## 
## 2. Quantiles for each variable:
## 
##            2.5%    25%    50%    75%  97.5%
## period 1 -1.969 -1.618 -1.430 -1.262 -0.903
## period 2 -3.122 -2.609 -2.395 -2.165 -1.746
## period 3 -3.199 -2.739 -2.516 -2.286 -1.870
## period 4 -3.890 -3.254 -2.968 -2.698 -2.003Note that as.mcmc will actually produce an mcmc.list when there is more than one chain present, as in this example. The 2.5th and 97.5th quantiles are similar, but not identical, to the 95% confidence intervals in the frequentist summary. Here is a plot of the posterior EMMs, back-transformed:
bayesplot::mcmc_areas(as.mcmc(regrid(period.emmGrid)))… and here are intervals for each period compared with its neighbor; we wrap the important part of the call in an extra as.mcmc(as.matrix(...)) so as to combine the two chains into one.
HPDinterval(as.mcmc(as.matrix(
    as.mcmc(contrast(period.emmGrid, "consec", reverse = TRUE), names = FALSE))))
##            lower     upper
## 1 - 2  0.3749803 1.5329574
## 2 - 3 -0.5813314 0.9415829
## 3 - 4 -0.4765553 1.6714750
## attr(,"Probability")
## [1] 0.95The only interval that excludes zero is the one that compares periods 1 and 2.
In summary, to do Bayesian analysis in the emmeans package, use the same tools that are available for other models, extract the MCMC samples using as.mcmc(), and summarize or plot from there.