Logistic Regression

Last updated on 2025-01-21 | Edit this page

Estimated time: 120 minutes

Overview

Questions

  • How can I identify factors for antibiotic resistance?
  • How can I check the validity of model?

Objectives

  • To be able to construct regression models for a binary outcome
  • To be able to calculate predicted variables and residuals
  • To be able to present model outcomes using Broom

Content


  • Exploratory Data Analysis
  • Model Creation and Estimation
  • Reporting the Logistic Regression Results

Data


R

# We will need these libraries and this data later.
library(aod)
library(broom)
library(ggplot2)
library(lubridate)
library(odds.n.ends)
library(tidyverse)

amrData <- read.csv("data/dig_health_hub_amr.csv")

The data used in this episode was provided by Simon Thelwall from the UKHSA. It has been created to represent the sort of data that might be obtained from the Second Generation Surveillance System (SGSS). The data has 100,000 rows of 12 variables.

Exploratory Data Analysis


We can preview our data by using ‘head’:

R

head(amrData)

OUTPUT

  id        dob  spec_date sex_male     region had_surgery_past_yr
1  1 1981-04-04 2014-05-25        1     London                   0
2  2 1969-05-22 2014-05-25        0       East                   0
3  3 1935-11-20 2014-05-25        0     London                   1
4  4 1967-05-19 2014-05-25        1     London                   0
5  5 1961-06-30 2014-05-25        0 North West                   0
6  6 1987-01-11 2014-05-25        1 North East                   0
                                                ethnicity imd organism coamox
1                                      Other ethnic group   4  E. coli      1
2                                      Other ethnic group   3  E. coli      0
3 Black, Black British, Black Welsh, Caribbean or African   2  E. coli      1
4                         Mixed or Multiple ethnic groups   5  E. coli      0
5                         Mixed or Multiple ethnic groups   4  E. coli      0
6                         Mixed or Multiple ethnic groups   4  E. coli      0
  cipro gentam
1     0      0
2     0      0
3     0      0
4     0      0
5     0      0
6     0      0

We can also request information about variable names and data types:

R

sapply(amrData, class)

OUTPUT

                 id                 dob           spec_date            sex_male
          "integer"         "character"         "character"           "integer"
             region had_surgery_past_yr           ethnicity                 imd
        "character"           "integer"         "character"           "integer"
           organism              coamox               cipro              gentam
        "character"           "integer"           "integer"           "integer" 

We can see that dates are currently stored as the char data type. We also do not know the age of the participant when the sample was taken.

R

# Calculate age (in years) as of their last birthday and add as an additional variable to our data.
# The %--% and %/% are synax specific to lubridate. 
# In the first part we are asking it to find the difference between the two dates.
# We are then rounding down to the nearest year.

amrData <- amrData %>%
  mutate(
    age_years_sd = (dob %--% spec_date) %/% years(1)
  )

We can also convert ‘spec_date’, the date the specimen was taken from text to a date:

R

# Convert char to date and store as additional variable
amrData <- amrData %>%
  mutate(
    spec_date_YMD = as.Date(amrData$spec_date)
  )

We can use a histogram to explore the age distribution of the participants:

R

# histogram of age
ageHisto <- amrData %>%
  ggplot(aes(x = age_years_sd)) +
  geom_histogram(bins = 10, color = "white") +
  theme_minimal(base_size = 14, base_family = "sans") +
  labs(x = "Age when specimen taken (years)", y = "Frequency")
ageHisto

We can also look at where the specimens were processed:

R

xtabs(~region, data = amrData)

OUTPUT

region
                    East            East Midlands                   London
                   11084                    11160                    11253
              North East               North West               South East
                   11082                    10919                    11101
              South West            West Midlands Yorkshire and The Humber
                   11278                    11032                    11091 

and for which organism:

R

xtabs(~organism, data = amrData)

OUTPUT

organism
E. coli
 100000 

In addition, we can use cross-tabulation to identify if the specimen indicated resistance to one or more of Coamoxiclav, Gentamicin and Ciprofloxacin for the participants:

R

xtabs(~ coamox + cipro + gentam, data = amrData)

OUTPUT

, , gentam = 0

      cipro
coamox     0     1
     0 61844  5814
     1 24319  3109

, , gentam = 1

      cipro
coamox     0     1
     0  2815   362
     1  1510   227

We can see from our table that only 227 participants indicated reistance to Coamoxiclav, Gentamicin and Ciprofloxacin. Coamoxiclav appears to have the highest individual indication of resistance. We will explore indicators to Coamoxiclav first.

Model Creation and Estimation


As the dependent variable we want to explore is binary (0,1), we will use a binomial generalised linear model.

R

coamox_logit <- glm(coamox ~ age_years_sd + sex_male, data = amrData, family = "binomial")

summary(coamox_logit)

OUTPUT


Call:
glm(formula = coamox ~ age_years_sd + sex_male, family = "binomial",
    data = amrData)

Coefficients:
               Estimate Std. Error  z value Pr(>|z|)
(Intercept)  -3.2215791  0.0284626 -113.186  < 2e-16 ***
age_years_sd  0.0404280  0.0004449   90.867  < 2e-16 ***
sex_male      0.0540494  0.0146011    3.702 0.000214 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 120724  on 99999  degrees of freedom
Residual deviance: 111460  on 99997  degrees of freedom
AIC: 111466

Number of Fisher Scoring iterations: 4

In this initial model focusing on demographic indicators, age at time of sample being taken and whether or not the participant is male both seem to be statistically significant.

age_years_sd: For every unit increase in age_years_sd the log-odds of Coamoxiclav resistance increase by 0.0404280.

sex_male: The difference in the log-odds of Coamoxiclav resistance between males and non-males is 0.0540494.

Older participants and male participants have higher log-odds of Coamoxicalv resistance.

We can check to see that our indicators sex_male and age_years_sd are independent:

R

# check VIF for no perfect multicollinearity assumption
car::vif(coamox_logit)

OUTPUT

age_years_sd     sex_male
    1.000015     1.000015 

We can also check the linearity of the variable age_years_sd

R

# make a variable of the logit of the predicted values
logit.use <- log(coamox_logit$fitted.values / (1 - coamox_logit$fitted.values))

# make a small data frame with the logit variable and the age predictor
linearity.data <- data.frame(logit.use, age = coamox_logit$model$age_years_sd)

# create a plot with linear and actual relationships shown
linearPlot <- linearity.data %>%
  ggplot(aes(x = age, y = logit.use)) +
  geom_point(aes(size = "Observation"), color = "blue", alpha = .6) +
  geom_smooth(se = FALSE, aes(color = "Loess curve")) +
  geom_smooth(method = lm, se = FALSE, aes(color = "linear")) +
  theme_minimal(base_size = 14, base_family = "serif") +
  labs(x = "Age in years on sample date", y = "Log-odds of coamox resistance predicted probability") +
  scale_color_manual(name = "Type of fit line", values = c("red", "black")) +
  scale_size_manual(values = 1.5, name = "")

linearPlot

OUTPUT

`geom_smooth()` using method = 'gam' and formula = 'y ~ s(x, bs = "cs")'
`geom_smooth()` using formula = 'y ~ x'

Challenge 1

Update the model coamox_logit to include had_surgery_past_yr as an independent variable. What is the log-odds reported and is it statistically significant?

You may choose to create a new glm:

R

coamox_surg_logit <- glm(coamox ~ age_years_sd + sex_male + had_surgery_past_yr, data = amrData, family = "binomial")

summary(coamox_surg_logit)

OUTPUT


Call:
glm(formula = coamox ~ age_years_sd + sex_male + had_surgery_past_yr,
    family = "binomial", data = amrData)

Coefficients:
                      Estimate Std. Error  z value Pr(>|z|)
(Intercept)         -3.2407115  0.0286059 -113.288  < 2e-16 ***
age_years_sd         0.0404424  0.0004451   90.870  < 2e-16 ***
sex_male             0.0538298  0.0146056    3.686 0.000228 ***
had_surgery_past_yr  0.1802123  0.0238522    7.555 4.18e-14 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 120724  on 99999  degrees of freedom
Residual deviance: 111404  on 99996  degrees of freedom
AIC: 111412

Number of Fisher Scoring iterations: 4

had_surgery_past_yr: The difference in the log-odds of Coamoxiclav resistance between those who have had surgery in the past year and those who have not is 0.1802123. It is statistically significant.

You may also want to check for culticollinearity:

R

car::vif(coamox_surg_logit)

OUTPUT

       age_years_sd            sex_male had_surgery_past_yr
           1.000156            1.000018            1.000144 

As the value of GVIF is lower than 4, it suggests that the assumption of independnce between the variables is held.

Incorporating a multi-level factor

So far we have considered binary and continuos variable indicators in our model(s). Our data set also contains some categorical variables: region, ethinicity and imd.

We are going to incorporate region into our model:

R

# To include reporting for all regions I will include the 0 flag
coamox_region_logit <- glm(coamox ~ 0 + age_years_sd + sex_male + region, data = amrData, family = "binomial")
summary(coamox_region_logit)

OUTPUT


Call:
glm(formula = coamox ~ 0 + age_years_sd + sex_male + region,
    family = "binomial", data = amrData)

Coefficients:
                                Estimate Std. Error z value Pr(>|z|)
age_years_sd                    0.040432   0.000445  90.867  < 2e-16 ***
sex_male                        0.054155   0.014602   3.709 0.000208 ***
regionEast                     -3.246673   0.035328 -91.900  < 2e-16 ***
regionEast Midlands            -3.213569   0.035077 -91.616  < 2e-16 ***
regionLondon                   -3.204244   0.034972 -91.625  < 2e-16 ***
regionNorth East               -3.219060   0.035140 -91.606  < 2e-16 ***
regionNorth West               -3.217271   0.035244 -91.285  < 2e-16 ***
regionSouth East               -3.183229   0.035058 -90.799  < 2e-16 ***
regionSouth West               -3.222338   0.035057 -91.917  < 2e-16 ***
regionWest Midlands            -3.236012   0.035275 -91.736  < 2e-16 ***
regionYorkshire and The Humber -3.255309   0.035302 -92.213  < 2e-16 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 138629  on 100000  degrees of freedom
Residual deviance: 111452  on  99989  degrees of freedom
AIC: 111474

Number of Fisher Scoring iterations: 4

It initially seems that all regions are significant and have a negative log-odds of Coamoxiclav resistance.

We can use the Wald test to explore the overall effect of region. We will use the wald.test function of the aod library. The order in which the coefficients are given in the table of coefficients is the same as the order of the terms in the model. This is important because the wald.test function refers to the coefficients by their order in the model. We use the wald.test function. b supplies the coefficients, while Sigma supplies the variance covariance matrix of the error terms, finally Terms tells R which terms in the model are to be tested, in this case, terms 3 to 11, are the three terms for the different regions.

R

# model with just the regions added
wald.test(b = coef(coamox_region_logit), Sigma = vcov(coamox_region_logit), Terms = 3:11)

OUTPUT

Wald test:
----------

Chi-squared test:
X2 = 12815.9, df = 9, P(> X2) = 0.0

The chi-squared test statistic of 12815.9, with nine degrees of freedom is associated with a p-value of 0.00 indicating that the overall effect of region is statistically significant.

Challenge 2

Update the model you created in Challenge 1 to include either ethnicity or imd as an independent variable. What is the log-odds reported and is it statistically significant?

If you chose to incorporate imd, this would first need to be converted to a factor

R

amrData$imd <- factor(amrData$imd)

coamox_surg_imd_logit <- glm(
  coamox ~ age_years_sd + sex_male + had_surgery_past_yr + imd,
  data = amrData,
  family = "binomial"
)

summary(coamox_surg_imd_logit)

OUTPUT


Call:
glm(formula = coamox ~ age_years_sd + sex_male + had_surgery_past_yr +
    imd, family = "binomial", data = amrData)

Coefficients:
                      Estimate Std. Error  z value Pr(>|z|)
(Intercept)         -3.2405156  0.0320951 -100.966  < 2e-16 ***
age_years_sd         0.0404419  0.0004451   90.866  < 2e-16 ***
sex_male             0.0538963  0.0146067    3.690 0.000224 ***
had_surgery_past_yr  0.1801131  0.0238532    7.551 4.32e-14 ***
imd2                -0.0135384  0.0231010   -0.586 0.557840
imd3                -0.0044812  0.0230452   -0.194 0.845821
imd4                -0.0046000  0.0230556   -0.200 0.841857
imd5                 0.0216405  0.0230414    0.939 0.347627
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 120724  on 99999  degrees of freedom
Residual deviance: 111401  on 99992  degrees of freedom
AIC: 111417

Number of Fisher Scoring iterations: 4

As we did not include the 0 flag IMD quintile 1 has been used a reference for the other Quintiles. Quintiles 2 to 5 are not significantly significant. We may want to test that the coefficient for imd=1 is equal to the coefficient for imd=2. The first line of code below creates a vector l that defines the test we want to perform. In this case, we want to test the difference (subtraction) of the terms for imd=1 and imd=2 (i.e., the 4th and 5th terms in the model). To contrast these two terms, we multiply one of them by 1, and the other by -1. The other terms in the model are not involved in the test, so they are multiplied by 0. The second line of code below uses L=l to tell R that we wish to base the test on the vector l (rather than using the Terms option as we did above).

R

l <- cbind(0, 0, 0, 1, -1, 0, 0, 0)
wald.test(b = coef(coamox_surg_imd_logit), Sigma = vcov(coamox_surg_imd_logit), L = l)

OUTPUT

Wald test:
----------

Chi-squared test:
X2 = 33.9, df = 1, P(> X2) = 5.7e-09

The chi-squared test statistic of 33.9, with 1 degree of freedom is associated with a p-value of 5.7e-09 indicating that the overall effect of region is statistically significant.

Reporting the Logistic Regression Results


For our model coamox_region_logit there are various things that we can report, and different functions and packages that can be used.

We can report the CIs using profiled log-likelihood or using standard errors for our model.

R

# CIs using log-likelihood
confint(coamox_region_logit)

OUTPUT

Waiting for profiling to be done...

OUTPUT

                                     2.5 %      97.5 %
age_years_sd                    0.03956087  0.04130508
sex_male                        0.02553710  0.08277680
regionEast                     -3.31607182 -3.17758553
regionEast Midlands            -3.28247264 -3.14497346
regionLondon                   -3.27293944 -3.13585237
regionNorth East               -3.28808726 -3.15033821
regionNorth West               -3.28650481 -3.14834796
regionSouth East               -3.25209244 -3.11466610
regionSouth West               -3.29120225 -3.15377971
regionWest Midlands            -3.30530677 -3.16702900
regionYorkshire and The Humber -3.32465587 -3.18627353

R

## CIs using standard errors
confint.default(coamox_region_logit)

OUTPUT

                                     2.5 %      97.5 %
age_years_sd                    0.03955954  0.04130374
sex_male                        0.02553490  0.08277421
regionEast                     -3.31591576 -3.17743094
regionEast Midlands            -3.28231824 -3.14482054
regionLondon                   -3.27278672 -3.13570109
regionNorth East               -3.28793328 -3.15018570
regionNorth West               -3.28634826 -3.14819290
regionSouth East               -3.25194196 -3.11451707
regionSouth West               -3.29104882 -3.15362773
regionWest Midlands            -3.30515053 -3.16687424
regionYorkshire and The Humber -3.32449918 -3.18611831

Alternatively we may be interested in the odds-ratio:

R

## odds ratios only
exp(coef(coamox_region_logit))

OUTPUT

                  age_years_sd                       sex_male
                    1.04126013                     1.05564775
                    regionEast            regionEast Midlands
                    0.03890341                     0.04021282
                  regionLondon               regionNorth East
                    0.04058958                     0.03999265
              regionNorth West               regionSouth East
                    0.04006426                     0.04145157
              regionSouth West            regionWest Midlands
                    0.03986174                     0.03932038
regionYorkshire and The Humber
                    0.03856891 

R

# including confidence intervals
exp(cbind(OR = coef(coamox_region_logit), confint(coamox_region_logit)))

OUTPUT

Waiting for profiling to be done...

OUTPUT

                                       OR      2.5 %     97.5 %
age_years_sd                   1.04126013 1.04035382 1.04217000
sex_male                       1.05564775 1.02586597 1.08629932
regionEast                     0.03890341 0.03629513 0.04168618
regionEast Midlands            0.04021282 0.03753533 0.04306807
regionLondon                   0.04058958 0.03789487 0.04346269
regionNorth East               0.03999265 0.03732517 0.04283764
regionNorth West               0.04006426 0.03738429 0.04292298
regionSouth East               0.04145157 0.03869316 0.04439333
regionSouth West               0.03986174 0.03720909 0.04269046
regionWest Midlands            0.03932038 0.03668796 0.04212858
regionYorkshire and The Humber 0.03856891 0.03598490 0.04132558

These can also be presented using broom:

R

# as Log-Odds
tidy(coamox_region_logit)

OUTPUT

# A tibble: 11 × 5
   term                           estimate std.error statistic  p.value
   <chr>                             <dbl>     <dbl>     <dbl>    <dbl>
 1 age_years_sd                     0.0404  0.000445     90.9  0
 2 sex_male                         0.0542  0.0146        3.71 0.000208
 3 regionEast                      -3.25    0.0353      -91.9  0
 4 regionEast Midlands             -3.21    0.0351      -91.6  0
 5 regionLondon                    -3.20    0.0350      -91.6  0
 6 regionNorth East                -3.22    0.0351      -91.6  0
 7 regionNorth West                -3.22    0.0352      -91.3  0
 8 regionSouth East                -3.18    0.0351      -90.8  0
 9 regionSouth West                -3.22    0.0351      -91.9  0
10 regionWest Midlands             -3.24    0.0353      -91.7  0
11 regionYorkshire and The Humber  -3.26    0.0353      -92.2  0       

R

# as ORs
tidy(coamox_region_logit, exp = TRUE)

OUTPUT

# A tibble: 11 × 5
   term                           estimate std.error statistic  p.value
   <chr>                             <dbl>     <dbl>     <dbl>    <dbl>
 1 age_years_sd                     1.04    0.000445     90.9  0
 2 sex_male                         1.06    0.0146        3.71 0.000208
 3 regionEast                       0.0389  0.0353      -91.9  0
 4 regionEast Midlands              0.0402  0.0351      -91.6  0
 5 regionLondon                     0.0406  0.0350      -91.6  0
 6 regionNorth East                 0.0400  0.0351      -91.6  0
 7 regionNorth West                 0.0401  0.0352      -91.3  0
 8 regionSouth East                 0.0415  0.0351      -90.8  0
 9 regionSouth West                 0.0399  0.0351      -91.9  0
10 regionWest Midlands              0.0393  0.0353      -91.7  0
11 regionYorkshire and The Humber   0.0386  0.0353      -92.2  0       

The odd.n.ends package provides a wide range of reporting tools

R

# Odds ratios and confidence intervals
coamox_region_logitOR <- odds.n.ends::odds.n.ends(coamox_region_logit)

OUTPUT

Waiting for profiling to be done...

R

coamox_region_logitCI <- coamox_region_logitOR$`Predictor odds ratios and 95% CI`
coamox_region_logitCI

OUTPUT

                                       OR      2.5 %     97.5 %
age_years_sd                   1.04126013 1.04035382 1.04217000
sex_male                       1.05564775 1.02586597 1.08629932
regionEast                     0.03890341 0.03629513 0.04168618
regionEast Midlands            0.04021282 0.03753533 0.04306807
regionLondon                   0.04058958 0.03789487 0.04346269
regionNorth East               0.03999265 0.03732517 0.04283764
regionNorth West               0.04006426 0.03738429 0.04292298
regionSouth East               0.04145157 0.03869316 0.04439333
regionSouth West               0.03986174 0.03720909 0.04269046
regionWest Midlands            0.03932038 0.03668796 0.04212858
regionYorkshire and The Humber 0.03856891 0.03598490 0.04132558

R

# model fit
modfit <- coamox_region_logitOR$`Count R-squared (model fit): percent correctly predicted`
modfit

OUTPUT

[1] 70.938

R

# Other model statistics
odds.n.ends::odds.n.ends(coamox_region_logit)

OUTPUT

Waiting for profiling to be done...

OUTPUT

$`Logistic regression model significance`
Chi-squared        d.f.           p
  27177.424          11       <.001

$`Contingency tables (model fit): frequency predicted`
                Number observed
Number predicted      1      0    Sum
             1     4613   4510   9123
             0    24552  66325  90877
             Sum  29165  70835 100000

$`Count R-squared (model fit): percent correctly predicted`
[1] 70.938

$`Model sensitivity`
[1] 0.158169

$`Model specificity`
[1] 0.9363309

$`Predictor odds ratios and 95% CI`
                                       OR      2.5 %     97.5 %
age_years_sd                   1.04126013 1.04035382 1.04217000
sex_male                       1.05564775 1.02586597 1.08629932
regionEast                     0.03890341 0.03629513 0.04168618
regionEast Midlands            0.04021282 0.03753533 0.04306807
regionLondon                   0.04058958 0.03789487 0.04346269
regionNorth East               0.03999265 0.03732517 0.04283764
regionNorth West               0.04006426 0.03738429 0.04292298
regionSouth East               0.04145157 0.03869316 0.04439333
regionSouth West               0.03986174 0.03720909 0.04269046
regionWest Midlands            0.03932038 0.03668796 0.04212858
regionYorkshire and The Humber 0.03856891 0.03598490 0.04132558

As you can see we have a range of probabilities, odds ratios and log-odds. We need to be aware of which we are referring to. One way is to look at the range of the estimates. Probabilities always have a range from zero to 1. Logit units generally range from about -4 to +4, with zero meaning an equal probability of no event or the event outcome occurring. Odds ratios can range from very small (but positive) numbers to very large positive numbers.

These odds ratios versions of the estimates are more easily interpretable than logit scores. Odds ratios of less than one means that an increase in that predictor makes the outcome less likely to occur, and an odds ratio of greater than one means that an increase in that predictor makes the outcome more likely to occur.

Discussion

  • What other indicators could we have included in our model?
  • What question(s) would they help us to answer?

Challenge 3

We have explored potential indicators of Coamoxiclav resistance.

Explore potential indicators of either Gentamicin or Ciprofloxacin resistance.

Present your analysis steps and findings in a Quarto report.