Logistic Regression in R

In binary logistic regression, there is a single binary dependent variable, coded by an indicator variable. For example, if we represent a response as 1 and non-response as 0, then the corresponding probability of response, can be between 0 (certainly not a response) and 1 (certainly a response) - hence the labeling !

The logistic model models the log-odds of an event as a linear combination of one or more independent variables (explanatory variables). If we observed \((y_i, x_i),\) where \(y_i\) is a Bernoulli variable and \(x_i\) a vector of explanatory variables, the model for \(\pi_i = P(y_i=1)\) is

\[ \text{logit}(\pi_i)= \log\left\{ \frac{\pi_i}{1-\pi_i}\right\} = \beta_0 + \beta x_i, i = 1,\ldots,n \]

The model is especially useful in case-control studies and leads to the effect of risk factors by odds ratios.

Example: Lung Cancer Data

Data source: Loprinzi CL. Laurie JA. Wieand HS. Krook JE. Novotny PJ. Kugler JW. Bartel J. Law M. Bateman M. Klatt NE. et al. Prospective evaluation of prognostic variables from patient-completed questionnaires. North Central Cancer Treatment Group. Journal of Clinical Oncology. 12(3):601-7, 1994.

Survival in patients with advanced lung cancer from the North Central Cancer Treatment Group. Performance scores rate how well the patient can perform usual daily activities (see ?lung for details).

library(survival) 
glimpse(lung)
Rows: 228
Columns: 10
$ inst      <dbl> 3, 3, 3, 5, 1, 12, 7, 11, 1, 7, 6, 16, 11, 21, 12, 1, 22, 16…
$ time      <dbl> 306, 455, 1010, 210, 883, 1022, 310, 361, 218, 166, 170, 654…
$ status    <dbl> 2, 2, 1, 2, 2, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, …
$ age       <dbl> 74, 68, 56, 57, 60, 74, 68, 71, 53, 61, 57, 68, 68, 60, 57, …
$ sex       <dbl> 1, 1, 1, 1, 1, 1, 2, 2, 1, 1, 1, 2, 2, 1, 1, 1, 1, 1, 2, 1, …
$ ph.ecog   <dbl> 1, 0, 0, 1, 0, 1, 2, 2, 1, 2, 1, 2, 1, NA, 1, 1, 1, 2, 2, 1,…
$ ph.karno  <dbl> 90, 90, 90, 90, 100, 50, 70, 60, 70, 70, 80, 70, 90, 60, 80,…
$ pat.karno <dbl> 100, 90, 90, 60, 90, 80, 60, 80, 80, 70, 80, 70, 90, 70, 70,…
$ meal.cal  <dbl> 1175, 1225, NA, 1150, NA, 513, 384, 538, 825, 271, 1025, NA,…
$ wt.loss   <dbl> NA, 15, 15, 11, 0, 0, 10, 1, 16, 34, 27, 23, 5, 32, 60, 15, …

Model Fit

We analyze the event of weight gain (or staying the same weight) in lung cancer patients in dependency of age, sex, ECOG performance score and calories consumed at meals. In the original data, a positive number for the wt.loss variable is a weight loss, negative number is a gain. We start by dichotomising the response such that a result >0 is a weight loss, <= weight gain and creating a factor variable wt_grp.

One of the most important things to remember is to ensure you tell R what your event is ! We want to model Events / Non-events, and hence your reference category for wt_grp dichotomous variable below is the weight loss level. Therefore, by telling R that your reference category is weight loss, you are effectively telling R that your Event = Weight Gain !

lung2 <- survival::lung %>% 
  mutate(
    wt_grp = factor(wt.loss > 0, labels = c("weight loss", "weight gain"))
  ) 

#specify that weight loss should be used as baseline level (i.e we want to model weight gain as the event)
lung2$wt_grp <- relevel(lung2$wt_grp, ref='weight loss')

m1 <- glm(wt_grp ~ age + sex + ph.ecog + meal.cal, data = lung2, family = binomial(link="logit"))
summary(m1)

Call:
glm(formula = wt_grp ~ age + sex + ph.ecog + meal.cal, family = binomial(link = "logit"), 
    data = lung2)

Coefficients:
              Estimate Std. Error z value Pr(>|z|)  
(Intercept)  3.2631673  1.6488207   1.979   0.0478 *
age         -0.0101717  0.0208107  -0.489   0.6250  
sex         -0.8717357  0.3714042  -2.347   0.0189 *
ph.ecog      0.4179665  0.2588653   1.615   0.1064  
meal.cal    -0.0008869  0.0004467  -1.985   0.0471 *
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 202.36  on 169  degrees of freedom
Residual deviance: 191.50  on 165  degrees of freedom
  (58 observations deleted due to missingness)
AIC: 201.5

Number of Fisher Scoring iterations: 4

The model summary contains the parameter estimates \(\beta_j\) for each explanatory variable \(x_j\), corresponding to the log-odds for the response variable to take the value \(1\), conditional on all other explanatory variables remaining constant. For better interpretation, we can exponentiate these estimates, to obtain estimates for the odds instead and provide 95% confidence intervals:

exp(coef(m1))
(Intercept)         age         sex     ph.ecog    meal.cal 
 26.1321742   0.9898798   0.4182250   1.5188698   0.9991135 
exp(confint(m1))
Waiting for profiling to be done...
                2.5 %      97.5 %
(Intercept) 1.0964330 730.3978786
age         0.9495388   1.0307216
sex         0.1996925   0.8617165
ph.ecog     0.9194053   2.5491933
meal.cal    0.9982107   0.9999837
#to output to a single tibble
out1<-as_tibble(cbind(names(m1$coefficients),exp(cbind(Odds_Ratio = coef(m1), confint(m1)))))
Waiting for profiling to be done...
Warning: The `x` argument of `as_tibble.matrix()` must have unique column names if
`.name_repair` is omitted as of tibble 2.0.0.
ℹ Using compatibility `.name_repair`.
out1
# A tibble: 5 × 4
  V1          Odds_Ratio        `2.5 %`           `97.5 %`         
  <chr>       <chr>             <chr>             <chr>            
1 (Intercept) 26.132174204934   1.09643300839377  730.397878585508 
2 age         0.98987981211002  0.949538773158863 1.0307215976617  
3 sex         0.418224997956518 0.199692519608937 0.86171649021045 
4 ph.ecog     1.51886984334321  0.919405326114938 2.54919332383799 
5 meal.cal    0.999113450548666 0.99821071019642  0.999983731220538

NOTE: that the confidence intervals are being calculated using the profile likelihood method. See here for more details.

Model Comparison

To compare two logistic models, the residual deviances (-2 * log likelihoods) are compared against a \(\chi^2\)-distribution with degrees of freedom calculated using the difference in the two models’ parameters. Below, the only difference is the inclusion/exclusion of age in the model, hence we test using \(\chi^2\) with 1 df. Here testing at the 5% level.

m2 <- glm(wt_grp ~ sex + ph.ecog + meal.cal, data = lung2, family = binomial(link="logit"))
summary(m2)

Call:
glm(formula = wt_grp ~ sex + ph.ecog + meal.cal, family = binomial(link = "logit"), 
    data = lung2)

Coefficients:
              Estimate Std. Error z value Pr(>|z|)   
(Intercept)  2.5606595  0.7976887   3.210  0.00133 **
sex         -0.8359241  0.3637378  -2.298  0.02155 * 
ph.ecog      0.3794295  0.2469030   1.537  0.12435   
meal.cal    -0.0008334  0.0004346  -1.918  0.05517 . 
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 202.36  on 169  degrees of freedom
Residual deviance: 191.74  on 166  degrees of freedom
  (58 observations deleted due to missingness)
AIC: 199.74

Number of Fisher Scoring iterations: 4
anova(m1, m2, test = "LRT")
Analysis of Deviance Table

Model 1: wt_grp ~ age + sex + ph.ecog + meal.cal
Model 2: wt_grp ~ sex + ph.ecog + meal.cal
  Resid. Df Resid. Dev Df Deviance Pr(>Chi)
1       165     191.50                     
2       166     191.75 -1 -0.24046   0.6239

Stackexchange here has a good article describing this method and the difference between comparing 2 models using the likelihood ratio tests versus using wald tests and Pr>chisq (from the maximum likelihood estimate). Note: anova(m1, m2, test = "Chisq") and using test="LRT" as above are synonymous in this context.

Prediction

Predictions from the model for the log-odds of a patient with new data to experience a weight loss are derived using predict():

# new female, symptomatic but completely ambulatory patient consuming 2500 calories
new_pt <- data.frame(sex=2, ph.ecog=1, meal.cal=2500)
predict(m2, new_pt, type = "response")
       1 
0.306767 

Contrast statements for 2 or more treatments

To create contrasts, you can use the fit.contrast() function from the gmodels package.

This can be used with lm and glm objections:

Suppose we had a 3 level treatment variable (trt01p), whose levels were ordered Dose1, Dose2, Placebo.

You would fit the model as above, followed by fit.contrast(). This is effective testing the null hypothesis that 0.5dose1 + 0.5 dose2 - placebo = 0.

m2 <- glm(wt_grp ~ sex + trt01p, data = lung2, family = binomial(link=“logit”))

fit.contrast(m2,‘trt01p’,c(0.5,0.5,-1),conf.int=0.95)

Reference

#\| echo: false

#List all the packages needed

si <- sessioninfo::session_info(c('tidyverse','survival')) 
si
─ Session info ───────────────────────────────────────────────────────────────
 setting  value
 version  R version 4.4.2 (2024-10-31)
 os       Ubuntu 24.04.1 LTS
 system   x86_64, linux-gnu
 ui       X11
 language (EN)
 collate  C.UTF-8
 ctype    C.UTF-8
 tz       UTC
 date     2025-01-20
 pandoc   3.4 @ /opt/quarto/bin/tools/ (via rmarkdown)

─ Packages ───────────────────────────────────────────────────────────────────
 ! package       * version date (UTC) lib source
   askpass         1.2.1   2024-10-04 [1] RSPM (R 4.4.0)
   backports       1.5.0   2024-05-23 [1] RSPM (R 4.4.0)
   base64enc       0.1-3   2015-07-28 [1] RSPM (R 4.4.0)
   bit             4.5.0   2024-09-20 [1] RSPM (R 4.4.0)
   bit64           4.5.2   2024-09-22 [1] RSPM (R 4.4.0)
   blob            1.2.4   2023-03-17 [1] RSPM (R 4.4.0)
   broom           1.0.7   2024-09-26 [1] RSPM (R 4.4.0)
   bslib           0.8.0   2024-07-29 [1] RSPM (R 4.4.0)
   cachem          1.1.0   2024-05-16 [1] RSPM (R 4.4.0)
   callr           3.7.6   2024-03-25 [1] RSPM (R 4.4.0)
   cellranger      1.1.0   2016-07-27 [1] RSPM (R 4.4.0)
 P cli             3.6.3   2024-06-21 [?] RSPM (R 4.4.0)
   clipr           0.8.0   2022-02-22 [1] RSPM (R 4.4.0)
 P colorspace      2.1-1   2024-07-26 [?] RSPM (R 4.4.0)
   conflicted      1.2.0   2023-02-01 [1] RSPM (R 4.4.0)
   cpp11           0.5.0   2024-08-27 [1] RSPM (R 4.4.0)
   crayon          1.5.3   2024-06-20 [1] RSPM (R 4.4.0)
   curl            5.2.3   2024-09-20 [1] RSPM (R 4.4.0)
   data.table      1.16.0  2024-08-27 [1] RSPM (R 4.4.0)
   DBI             1.2.3   2024-06-02 [1] RSPM (R 4.4.0)
   dbplyr          2.5.0   2024-03-19 [1] RSPM (R 4.4.0)
 P digest          0.6.37  2024-08-19 [?] RSPM (R 4.4.0)
 P dplyr         * 1.1.4   2023-11-17 [?] RSPM (R 4.4.0)
   dtplyr          1.3.1   2023-03-22 [1] RSPM (R 4.4.0)
 P evaluate        1.0.0   2024-09-17 [?] RSPM (R 4.4.0)
 P fansi           1.0.6   2023-12-08 [?] RSPM (R 4.4.0)
   farver          2.1.2   2024-05-13 [1] RSPM (R 4.4.0)
 P fastmap         1.2.0   2024-05-15 [?] RSPM (R 4.4.0)
   fontawesome     0.5.2   2023-08-19 [1] RSPM (R 4.4.0)
 P forcats       * 1.0.0   2023-01-29 [?] RSPM (R 4.4.0)
   fs              1.6.4   2024-04-25 [1] RSPM (R 4.4.0)
   gargle          1.5.2   2023-07-20 [1] RSPM (R 4.4.0)
 P generics        0.1.3   2022-07-05 [?] RSPM (R 4.4.0)
 P ggplot2       * 3.5.1   2024-04-23 [?] RSPM (R 4.4.0)
 P glue            1.8.0   2024-09-30 [?] RSPM (R 4.4.0)
   googledrive     2.1.1   2023-06-11 [1] RSPM (R 4.4.0)
   googlesheets4   1.1.1   2023-06-11 [1] RSPM (R 4.4.0)
 P gtable          0.3.5   2024-04-22 [?] RSPM (R 4.4.0)
   haven           2.5.4   2023-11-30 [1] RSPM (R 4.4.0)
   highr           0.11    2024-05-26 [1] RSPM (R 4.4.0)
 P hms             1.1.3   2023-03-21 [?] RSPM (R 4.4.0)
 P htmltools       0.5.8.1 2024-04-04 [?] RSPM (R 4.4.0)
   httr            1.4.7   2023-08-15 [1] RSPM (R 4.4.0)
   ids             1.0.1   2017-05-31 [1] RSPM (R 4.4.0)
   isoband         0.2.7   2022-12-20 [1] RSPM (R 4.4.0)
   jquerylib       0.1.4   2021-04-26 [1] RSPM (R 4.4.0)
 P jsonlite        1.8.9   2024-09-20 [?] RSPM (R 4.4.0)
 P knitr           1.48    2024-07-07 [?] RSPM (R 4.4.0)
   labeling        0.4.3   2023-08-29 [1] RSPM (R 4.4.0)
   lattice         0.22-6  2024-03-20 [2] CRAN (R 4.4.2)
 P lifecycle       1.0.4   2023-11-07 [?] RSPM (R 4.4.0)
 P lubridate     * 1.9.3   2023-09-27 [?] RSPM (R 4.4.0)
 P magrittr        2.0.3   2022-03-30 [?] RSPM (R 4.4.0)
   MASS            7.3-61  2024-06-13 [2] CRAN (R 4.4.2)
   Matrix          1.7-1   2024-10-18 [2] CRAN (R 4.4.2)
   memoise         2.0.1   2021-11-26 [1] RSPM (R 4.4.0)
   mgcv            1.9-1   2023-12-21 [2] CRAN (R 4.4.2)
   mime            0.12    2021-09-28 [1] RSPM (R 4.4.0)
   modelr          0.1.11  2023-03-22 [1] RSPM (R 4.4.0)
 P munsell         0.5.1   2024-04-01 [?] RSPM (R 4.4.0)
   nlme            3.1-166 2024-08-14 [2] CRAN (R 4.4.2)
   openssl         2.2.2   2024-09-20 [1] RSPM (R 4.4.0)
 P pillar          1.9.0   2023-03-22 [?] RSPM (R 4.4.0)
 P pkgconfig       2.0.3   2019-09-22 [?] RSPM (R 4.4.0)
   prettyunits     1.2.0   2023-09-24 [1] RSPM (R 4.4.0)
   processx        3.8.4   2024-03-16 [1] RSPM (R 4.4.0)
   progress        1.2.3   2023-12-06 [1] RSPM (R 4.4.0)
   ps              1.8.0   2024-09-12 [1] RSPM (R 4.4.0)
 P purrr         * 1.0.2   2023-08-10 [?] RSPM (R 4.4.0)
 P R6              2.5.1   2021-08-19 [?] RSPM (R 4.4.0)
   ragg            1.3.3   2024-09-11 [1] RSPM (R 4.4.0)
   rappdirs        0.3.3   2021-01-31 [1] RSPM (R 4.4.0)
   RColorBrewer    1.1-3   2022-04-03 [1] RSPM (R 4.4.0)
 P readr         * 2.1.5   2024-01-10 [?] RSPM (R 4.4.0)
   readxl          1.4.3   2023-07-06 [1] RSPM (R 4.4.0)
   rematch         2.0.0   2023-08-30 [1] RSPM (R 4.4.0)
   rematch2        2.1.2   2020-05-01 [1] RSPM (R 4.4.0)
   reprex          2.1.1   2024-07-06 [1] RSPM (R 4.4.0)
 P rlang           1.1.4   2024-06-04 [?] RSPM (R 4.4.0)
 P rmarkdown       2.28    2024-08-17 [?] RSPM (R 4.4.0)
   rstudioapi      0.16.0  2024-03-24 [1] RSPM (R 4.4.0)
   rvest           1.0.4   2024-02-12 [1] RSPM (R 4.4.0)
   sass            0.4.9   2024-03-15 [1] RSPM (R 4.4.0)
 P scales          1.3.0   2023-11-28 [?] RSPM (R 4.4.0)
   selectr         0.4-2   2019-11-20 [1] RSPM (R 4.4.0)
 P stringi         1.8.4   2024-05-06 [?] RSPM (R 4.4.0)
 P stringr       * 1.5.1   2023-11-14 [?] RSPM (R 4.4.0)
   survival      * 3.7-0   2024-06-05 [2] CRAN (R 4.4.2)
   sys             3.4.3   2024-10-04 [1] RSPM (R 4.4.0)
   systemfonts     1.1.0   2024-05-15 [1] RSPM (R 4.4.0)
   textshaping     0.4.0   2024-05-24 [1] RSPM (R 4.4.0)
 P tibble        * 3.2.1   2023-03-20 [?] RSPM (R 4.4.0)
 P tidyr         * 1.3.1   2024-01-24 [?] RSPM (R 4.4.0)
 P tidyselect      1.2.1   2024-03-11 [?] RSPM (R 4.4.0)
 P tidyverse     * 2.0.0   2023-02-22 [?] RSPM (R 4.4.0)
 P timechange      0.3.0   2024-01-18 [?] RSPM (R 4.4.0)
   tinytex         0.53    2024-09-15 [1] RSPM (R 4.4.0)
 P tzdb            0.4.0   2023-05-12 [?] RSPM (R 4.4.0)
 P utf8            1.2.4   2023-10-22 [?] RSPM (R 4.4.0)
   uuid            1.2-1   2024-07-29 [1] RSPM (R 4.4.0)
 P vctrs           0.6.5   2023-12-01 [?] RSPM (R 4.4.0)
   viridisLite     0.4.2   2023-05-02 [1] RSPM (R 4.4.0)
   vroom           1.6.5   2023-12-05 [1] RSPM (R 4.4.0)
 P withr           3.0.1   2024-07-31 [?] RSPM (R 4.4.0)
 P xfun            0.48    2024-10-03 [?] RSPM (R 4.4.0)
   xml2            1.3.6   2023-12-04 [1] RSPM (R 4.4.0)
 P yaml            2.3.10  2024-07-26 [?] RSPM (R 4.4.0)

 [1] /home/runner/work/CAMIS/CAMIS/renv/library/linux-ubuntu-noble/R-4.4/x86_64-pc-linux-gnu
 [2] /opt/R/4.4.2/lib/R/library

 P ── Loaded and on-disk path mismatch.

──────────────────────────────────────────────────────────────────────────────