Linear regression and Broom
Last updated on 2025-01-21 | Edit this page
Overview
Questions
- How can I explore relationships between variables in my data?
- How can I present model outputs in an easier to read way?
Objectives
- To be able to explore relationships between variables
- To be able to calculate predicted variables and residuals
- To be able to construct linear regression models
- To be able to present model outcomes using Broom
Content
- Linear Regression Models
- Use of Log transform
- Use of Categorical Variables
- Use of Broom
Data
R
# We will need these libraries and this data later.
library(ggplot2)
library(tidyverse)
library(lmtest)
library(sandwich)
library(broom)
lon_dims_imd_2019 <- read.csv("data/English_IMD_2019_Domains_rebased_London_by_CDRC.csv")
We are going to use the data from the Consumer Data Research Centre, specifically the London IMD 2019 (English IMD 2019 Domains rebased).
Atribution: Data provided by the Consumer Data Research Centre, an ESRC Data Investment: ES/L011840/1, ES/L011891/1
The statistical unit areas across the country are Lower layer Super Output Areas (LSOAs). We will explore the relationships between the different dimensions of the Indices of Multiple Deprivation.
Linear regression
Linear Regression enables use to to explore the the linear relationship of the dependent variable Y and independent variable(s) X(s). We are going to explore the linear relationship between the Health Deprivation and Disability Domain and the Living Environment Deprivation Domain.
The Health Deprivation and Disability Domain measures the risk of premature death and the impairment of quality of life through poor physical or mental health. The domain measures morbidity, disability and premature mortality but not aspects of behaviour or environment that may be predictive of future health deprivation.
The Living Environment Deprivation Domain measures the quality of the local environment. The indicators fall into two sub-domains. The ‘indoors’ living environment measures the quality of housing; while the ‘outdoors’ living environment contains measures of air quality and road traffic accidents.
Reference: McLennan, David et al. The English Indices of Deprivation 2019 : Technical Report. Ministry of Housing, Communities and Local Government, 2019. Print.
Simple linear regression
In the simple linear regression example we have only one dependent variable (health_london_rank) and one independent variable (livingEnv_london_rank).
R
reg_LivEnv_health <- lm(health_london_rank ~ livingEnv_london_rank, data = lon_dims_imd_2019)
# We put the dependent variable to the left of the '~' and the independent variable(s) to the right
# and we tell R which dataset we are referring to.
summary(reg_LivEnv_health)
OUTPUT
Call:
lm(formula = health_london_rank ~ livingEnv_london_rank, data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-21549.6 -5948.2 609.1 6239.2 15792.2
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 1.692e+04 2.274e+02 74.40 <2e-16 ***
livingEnv_london_rank 3.430e-01 1.915e-02 17.91 <2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 7625 on 4833 degrees of freedom
Multiple R-squared: 0.06225, Adjusted R-squared: 0.06205
F-statistic: 320.8 on 1 and 4833 DF, p-value: < 2.2e-16
From the result of this analysis, we can see that the Living Environment Deprivation Domain rank has a significant(small p-value, general rule of thumb <0.05) and positive relationship(positive coefficient) with the Health Deprivation and Disability Domain rank.
One way of interpreting the result is: One unit increase in the Living Environment rank is related to around 0.343 (3.430e-01) points increase of the Health Deprivation and Disability rank.
R-square shows the amount of variance of Y explained by X. In this case the Living Environment rank explains 6.225% of the variance in the Health Deprivation and Disability rank. Adj R2(6.205%) shows the same as R2 but adjusted by the # of cases and # of variables. When the # of variables is small and the # of cases is very large then Adj R2 is closer to R2.
Log transform
If your data is skewed, it can be useful to transform a variable to it’s log form when doing the regression. You can either transform the variable beforehand or do so in the equation.
R
reg_logbarriers_health <- lm(health_london_rank ~ log(barriers_london_rank), data = lon_dims_imd_2019)
summary(reg_logbarriers_health)
OUTPUT
Call:
lm(formula = health_london_rank ~ log(barriers_london_rank),
data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-20379.3 -5611.7 774.9 5988.6 23828.5
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) -1349.4 798.3 -1.69 0.091 .
log(barriers_london_rank) 2917.0 105.7 27.59 <2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 7319 on 4833 degrees of freedom
Multiple R-squared: 0.1361, Adjusted R-squared: 0.1359
F-statistic: 761.1 on 1 and 4833 DF, p-value: < 2.2e-16
The interpretation of the log-transformed variable is a bit different. In this example only the predictor variable is log tranformed, therefore to interpret the slope coefficient we divide it by 100 (2917.0/100=29.170).
If the dependent/response variable is solely log-transformed, exponentiate the coefficient. This gives the multiplicative factor for every one-unit increase in the independent variable. Example: the coefficient is 0.198. exp(0.198) = 1.218962. For every one-unit increase in the independent variable, our dependent variable increases by a factor of about 1.22, or 22%. Recall that multiplying a number by 1.22 is the same as increasing the number by 22%. Likewise, multiplying a number by, say 0.84, is the same as decreasing the number by 1 – 0.84 = 0.16, or 16%.
If both are transformed, interpret the coefficient as the percent increase in the dependent variable for every 1% increase in the independent variable. Example: the coefficient is 0.198. For every 1% increase in the independent variable, our dependent variable increases by about 0.20%. For x percent increase, calculate 1.x to the power of the coefficient, subtract 1, and multiply by 100. Example: For every 20% increase in the independent variable, our dependent variable increases by about (1.20 0.198 - 1) * 100 = 3.7 percent.
Predicted values and residuals
We can expand our simple linear regression example to incorporate the Barriers to Housing and Services Domain rank. The Barriers to Housing and Services Domain measures the physical and financial accessibility of housing and local services. The indicators fall into two sub-domains: ‘geographical barriers’, which relate to the physical proximity of local services, and ‘wider barriers’ which includes issues relating to access to housing, such as affordability.
R
reg_LivEnv_barriers_health <- lm(
health_london_rank ~ livingEnv_london_rank + barriers_london_rank,
data = lon_dims_imd_2019
)
summary(reg_LivEnv_barriers_health)
OUTPUT
Call:
lm(formula = health_london_rank ~ livingEnv_london_rank + barriers_london_rank,
data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-21685.8 -4834.9 546.5 5142.4 18971.7
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 1.169e+04 2.502e+02 46.74 <2e-16 ***
livingEnv_london_rank 2.620e-01 1.721e-02 15.22 <2e-16 ***
barriers_london_rank 2.508e+00 7.059e-02 35.54 <2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 6790 on 4832 degrees of freedom
Multiple R-squared: 0.2566, Adjusted R-squared: 0.2562
F-statistic: 833.7 on 2 and 4832 DF, p-value: < 2.2e-16
After running the regression model, we can access the model predicted values and the residuals compared to the real observations.
R
# first we fit the predictions
health_rank_pred <- fitted(reg_LivEnv_barriers_health)
health_rank_pred <- as.data.frame(health_rank_pred)
# now we add the residual values too
health_rank_resid <- residuals(reg_LivEnv_barriers_health)
health_rank_pred$resid <- health_rank_resid
# We can thenview the predictions and residuals
head(health_rank_pred)
OUTPUT
health_rank_pred resid
1 20454.30 11658.702
2 24260.97 5444.031
3 15233.92 2366.080
4 16671.36 1235.645
5 15719.80 5861.196
6 14981.47 1432.528
R
# You can view the full data in RStudio with the View() function
View(health_rank_pred)
Robust regression
We can run the robust standard error regressions(control for heteroskedasticity, meaning unequal variances):
R
reg_LivEnv_barriers_health$robse <- vcovHC(reg_LivEnv_barriers_health, type = "HC1")
coeftest(reg_LivEnv_barriers_health, reg_LivEnv_barriers_health$robse)
OUTPUT
t test of coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 1.1694e+04 2.3776e+02 49.182 < 2.2e-16 ***
livingEnv_london_rank 2.6197e-01 1.7458e-02 15.006 < 2.2e-16 ***
barriers_london_rank 2.5085e+00 6.4394e-02 38.955 < 2.2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
In addition, we can access the cluster-robust standard errors regression results:
R
# cluster-robust standard errors
coeftest(reg_LivEnv_barriers_health, reg_LivEnv_barriers_health$clse)
OUTPUT
t test of coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 1.1694e+04 2.5018e+02 46.741 < 2.2e-16 ***
livingEnv_london_rank 2.6197e-01 1.7207e-02 15.225 < 2.2e-16 ***
barriers_london_rank 2.5085e+00 7.0588e-02 35.537 < 2.2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Challenge 1
Use the gapminder
data to create a linear model between
two continuous variables.
Discuss your question and your findings.
Regression with categorical independent variables
We will explore the use of categorical independent variables in linear regression in this episode. When the dependent variable is a categorical variable, you may consider the alternatives of linear regression like logit regression and multinomial regression.
R
# As a categorical variable we have added la19nm, these are the names of the London boroughs
reg_cat_var <- lm(health_london_rank ~ livingEnv_london_rank + barriers_london_rank + la19nm, data = lon_dims_imd_2019)
summary(reg_cat_var)
OUTPUT
Call:
lm(formula = health_london_rank ~ livingEnv_london_rank + barriers_london_rank +
la19nm, data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-21694.3 -3423.6 248.1 3691.8 19321.5
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 1.105e+04 5.429e+02 20.346 < 2e-16 ***
livingEnv_london_rank 3.595e-02 1.802e-02 1.996 0.046036 *
barriers_london_rank 3.149e+00 7.888e-02 39.924 < 2e-16 ***
la19nmBarnet 8.400e+03 6.508e+02 12.908 < 2e-16 ***
la19nmBexley 1.416e+03 7.186e+02 1.970 0.048856 *
la19nmBrent 7.765e+03 6.545e+02 11.863 < 2e-16 ***
la19nmBromley 5.208e+03 6.789e+02 7.670 2.06e-14 ***
la19nmCamden -1.842e+03 7.426e+02 -2.480 0.013184 *
la19nmCity of London 4.742e+03 2.251e+03 2.107 0.035185 *
la19nmCroydon 7.488e+02 6.388e+02 1.172 0.241192
la19nmEaling 3.697e+03 6.459e+02 5.724 1.11e-08 ***
la19nmEnfield 7.344e+03 6.501e+02 11.297 < 2e-16 ***
la19nmGreenwich -2.221e+03 6.873e+02 -3.232 0.001238 **
la19nmHackney -2.725e+03 6.811e+02 -4.000 6.43e-05 ***
la19nmHammersmith and Fulham -3.907e+03 7.438e+02 -5.252 1.57e-07 ***
la19nmHaringey 1.019e+03 6.877e+02 1.481 0.138580
la19nmHarrow 9.993e+03 7.053e+02 14.168 < 2e-16 ***
la19nmHavering -3.945e+02 7.317e+02 -0.539 0.589781
la19nmHillingdon 8.459e+02 6.937e+02 1.219 0.222791
la19nmHounslow 2.462e+03 6.890e+02 3.573 0.000356 ***
la19nmIslington -8.217e+03 7.305e+02 -11.248 < 2e-16 ***
la19nmKensington and Chelsea 9.191e+03 7.473e+02 12.299 < 2e-16 ***
la19nmKingston upon Thames 4.899e+03 7.804e+02 6.277 3.75e-10 ***
la19nmLambeth -6.270e+03 6.830e+02 -9.180 < 2e-16 ***
la19nmLewisham -1.991e+03 6.675e+02 -2.983 0.002869 **
la19nmMerton 3.387e+02 7.474e+02 0.453 0.650434
la19nmNewham 4.209e+03 6.617e+02 6.361 2.19e-10 ***
la19nmRedbridge 4.420e+03 6.914e+02 6.392 1.79e-10 ***
la19nmRichmond upon Thames 4.469e+03 7.740e+02 5.774 8.22e-09 ***
la19nmSouthwark -3.363e+03 6.731e+02 -4.996 6.05e-07 ***
la19nmSutton -1.729e+02 7.537e+02 -0.229 0.818563
la19nmTower Hamlets -5.862e+03 6.968e+02 -8.414 < 2e-16 ***
la19nmWaltham Forest 2.026e+03 6.854e+02 2.956 0.003137 **
la19nmWandsworth 1.087e+01 6.797e+02 0.016 0.987246
la19nmWestminster 5.683e+02 7.472e+02 0.761 0.446958
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 5362 on 4800 degrees of freedom
Multiple R-squared: 0.5395, Adjusted R-squared: 0.5363
F-statistic: 165.4 on 34 and 4800 DF, p-value: < 2.2e-16
R automatically recognizes la19nm as a factor and treats it accordingly. The missing one in the coefficient summary (Barking and Dagenham) is treated as a base line, therefore the value is 0. However, we can also modify our model to show for all:
R
reg_cat_var_showall <- lm(
health_london_rank ~ 0 + livingEnv_london_rank + barriers_london_rank + la19nm,
data = lon_dims_imd_2019
)
summary(reg_cat_var_showall)
OUTPUT
Call:
lm(formula = health_london_rank ~ 0 + livingEnv_london_rank +
barriers_london_rank + la19nm, data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-21694.3 -3423.6 248.1 3691.8 19321.5
Coefficients:
Estimate Std. Error t value Pr(>|t|)
livingEnv_london_rank 3.595e-02 1.802e-02 1.996 0.046 *
barriers_london_rank 3.149e+00 7.888e-02 39.924 < 2e-16 ***
la19nmBarking and Dagenham 1.105e+04 5.429e+02 20.346 < 2e-16 ***
la19nmBarnet 1.945e+04 4.764e+02 40.820 < 2e-16 ***
la19nmBexley 1.246e+04 5.876e+02 21.209 < 2e-16 ***
la19nmBrent 1.881e+04 4.526e+02 41.564 < 2e-16 ***
la19nmBromley 1.625e+04 5.444e+02 29.858 < 2e-16 ***
la19nmCamden 9.205e+03 5.783e+02 15.918 < 2e-16 ***
la19nmCity of London 1.579e+04 2.198e+03 7.184 7.78e-13 ***
la19nmCroydon 1.180e+04 4.574e+02 25.789 < 2e-16 ***
la19nmEaling 1.474e+04 4.388e+02 33.601 < 2e-16 ***
la19nmEnfield 1.839e+04 4.549e+02 40.431 < 2e-16 ***
la19nmGreenwich 8.825e+03 5.233e+02 16.865 < 2e-16 ***
la19nmHackney 8.322e+03 4.675e+02 17.802 < 2e-16 ***
la19nmHammersmith and Fulham 7.140e+03 5.675e+02 12.582 < 2e-16 ***
la19nmHaringey 1.207e+04 4.848e+02 24.886 < 2e-16 ***
la19nmHarrow 2.104e+04 5.667e+02 37.127 < 2e-16 ***
la19nmHavering 1.065e+04 6.178e+02 17.243 < 2e-16 ***
la19nmHillingdon 1.189e+04 5.541e+02 21.464 < 2e-16 ***
la19nmHounslow 1.351e+04 5.050e+02 26.750 < 2e-16 ***
la19nmIslington 2.830e+03 5.500e+02 5.145 2.78e-07 ***
la19nmKensington and Chelsea 2.024e+04 5.491e+02 36.859 < 2e-16 ***
la19nmKingston upon Thames 1.595e+04 6.476e+02 24.624 < 2e-16 ***
la19nmLambeth 4.777e+03 4.657e+02 10.257 < 2e-16 ***
la19nmLewisham 9.055e+03 4.543e+02 19.931 < 2e-16 ***
la19nmMerton 1.139e+04 5.935e+02 19.182 < 2e-16 ***
la19nmNewham 1.526e+04 4.467e+02 34.149 < 2e-16 ***
la19nmRedbridge 1.547e+04 5.307e+02 29.141 < 2e-16 ***
la19nmRichmond upon Thames 1.552e+04 6.355e+02 24.414 < 2e-16 ***
la19nmSouthwark 7.684e+03 4.662e+02 16.481 < 2e-16 ***
la19nmSutton 1.087e+04 6.274e+02 17.332 < 2e-16 ***
la19nmTower Hamlets 5.184e+03 5.107e+02 10.151 < 2e-16 ***
la19nmWaltham Forest 1.307e+04 4.831e+02 27.057 < 2e-16 ***
la19nmWandsworth 1.106e+04 4.905e+02 22.543 < 2e-16 ***
la19nmWestminster 1.161e+04 5.649e+02 20.560 < 2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 5362 on 4800 degrees of freedom
Multiple R-squared: 0.9407, Adjusted R-squared: 0.9403
F-statistic: 2177 on 35 and 4800 DF, p-value: < 2.2e-16
Categorical variables with interaction terms
Sometimes we are interested in how a variable interacts with another variable. We can explore any interactions between locations (la19nm) and the living environment and barrier ranks.
R
reg_cat_var_int <- lm(health_london_rank ~ la19nm * (livingEnv_london_rank + barriers_london_rank), data = lon_dims_imd_2019)
summary(reg_cat_var_int)
OUTPUT
Call:
lm(formula = health_london_rank ~ la19nm * (livingEnv_london_rank +
barriers_london_rank), data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-19227.9 -3254.4 297.6 3372.8 17891.5
Coefficients:
Estimate Std. Error
(Intercept) 1.300e+04 1.431e+03
la19nmBarnet 9.411e+03 1.909e+03
la19nmBexley -5.447e+03 2.464e+03
la19nmBrent 3.068e+03 1.866e+03
la19nmBromley 1.577e+03 2.107e+03
la19nmCamden -1.547e+04 3.565e+03
la19nmCity of London -8.405e+02 5.283e+03
la19nmCroydon -3.824e+03 1.679e+03
la19nmEaling 2.124e+03 1.841e+03
la19nmEnfield 2.420e+03 1.750e+03
la19nmGreenwich -3.383e+03 2.268e+03
la19nmHackney -1.803e+03 1.937e+03
la19nmHammersmith and Fulham -1.123e+04 2.389e+03
la19nmHaringey -4.527e+02 1.975e+03
la19nmHarrow 7.800e+03 2.476e+03
la19nmHavering -8.133e+03 2.519e+03
la19nmHillingdon 7.792e+02 2.030e+03
la19nmHounslow 4.939e+03 2.064e+03
la19nmIslington -4.614e+03 2.592e+03
la19nmKensington and Chelsea 1.264e+04 2.185e+03
la19nmKingston upon Thames 5.297e+03 2.972e+03
la19nmLambeth -6.973e+03 2.276e+03
la19nmLewisham -4.745e+03 1.912e+03
la19nmMerton -8.433e+03 2.793e+03
la19nmNewham 3.468e+03 1.786e+03
la19nmRedbridge 4.339e+03 2.156e+03
la19nmRichmond upon Thames 3.822e+03 4.288e+03
la19nmSouthwark -6.830e+03 1.923e+03
la19nmSutton -8.242e+03 2.600e+03
la19nmTower Hamlets -2.705e+03 2.163e+03
la19nmWaltham Forest 2.070e+03 1.814e+03
la19nmWandsworth 6.438e+02 2.060e+03
la19nmWestminster -1.063e+04 3.281e+03
livingEnv_london_rank -1.900e-01 1.321e-01
barriers_london_rank 3.620e+00 8.023e-01
la19nmBarnet:livingEnv_london_rank 1.641e-01 1.553e-01
la19nmBexley:livingEnv_london_rank 5.286e-01 1.644e-01
la19nmBrent:livingEnv_london_rank 5.219e-01 1.663e-01
la19nmBromley:livingEnv_london_rank 4.577e-01 1.492e-01
la19nmCamden:livingEnv_london_rank -4.599e-02 1.783e-01
la19nmCity of London:livingEnv_london_rank 3.448e-01 9.482e-01
la19nmCroydon:livingEnv_london_rank 4.524e-01 1.417e-01
la19nmEaling:livingEnv_london_rank 2.053e-01 1.624e-01
la19nmEnfield:livingEnv_london_rank 5.217e-01 1.556e-01
la19nmGreenwich:livingEnv_london_rank 2.230e-01 1.620e-01
la19nmHackney:livingEnv_london_rank -2.882e-01 1.817e-01
la19nmHammersmith and Fulham:livingEnv_london_rank 1.915e-01 2.025e-01
la19nmHaringey:livingEnv_london_rank -9.428e-02 2.048e-01
la19nmHarrow:livingEnv_london_rank 3.559e-01 1.774e-01
la19nmHavering:livingEnv_london_rank 5.238e-01 1.584e-01
la19nmHillingdon:livingEnv_london_rank 3.026e-01 1.572e-01
la19nmHounslow:livingEnv_london_rank -3.776e-02 1.691e-01
la19nmIslington:livingEnv_london_rank -4.471e-01 1.967e-01
la19nmKensington and Chelsea:livingEnv_london_rank -1.673e+00 2.725e-01
la19nmKingston upon Thames:livingEnv_london_rank 2.551e-01 1.748e-01
la19nmLambeth:livingEnv_london_rank -1.901e-01 2.037e-01
la19nmLewisham:livingEnv_london_rank 1.834e-01 1.687e-01
la19nmMerton:livingEnv_london_rank 5.430e-01 1.761e-01
la19nmNewham:livingEnv_london_rank -4.372e-03 1.627e-01
la19nmRedbridge:livingEnv_london_rank 2.310e-01 1.655e-01
la19nmRichmond upon Thames:livingEnv_london_rank -2.672e-03 1.876e-01
la19nmSouthwark:livingEnv_london_rank 2.786e-01 1.656e-01
la19nmSutton:livingEnv_london_rank 5.311e-01 1.576e-01
la19nmTower Hamlets:livingEnv_london_rank 1.954e-01 1.629e-01
la19nmWaltham Forest:livingEnv_london_rank -2.538e-02 1.729e-01
la19nmWandsworth:livingEnv_london_rank -1.552e-01 1.637e-01
la19nmWestminster:livingEnv_london_rank -9.646e-01 2.107e-01
la19nmBarnet:barriers_london_rank -1.334e+00 8.565e-01
la19nmBexley:barriers_london_rank -4.421e-01 8.661e-01
la19nmBrent:barriers_london_rank -7.477e-01 9.505e-01
la19nmBromley:barriers_london_rank -1.240e+00 8.666e-01
la19nmCamden:barriers_london_rank 2.994e+00 1.172e+00
la19nmCity of London:barriers_london_rank 1.001e+00 3.050e+00
la19nmCroydon:barriers_london_rank -6.200e-01 8.595e-01
la19nmEaling:barriers_london_rank -5.816e-01 8.604e-01
la19nmEnfield:barriers_london_rank -6.740e-01 8.776e-01
la19nmGreenwich:barriers_london_rank -8.003e-01 8.842e-01
la19nmHackney:barriers_london_rank 4.611e-01 1.427e+00
la19nmHammersmith and Fulham:barriers_london_rank 1.442e+00 9.425e-01
la19nmHaringey:barriers_london_rank 4.613e-01 9.154e-01
la19nmHarrow:barriers_london_rank -1.413e+00 9.202e-01
la19nmHavering:barriers_london_rank -3.766e-01 9.029e-01
la19nmHillingdon:barriers_london_rank -1.592e+00 8.683e-01
la19nmHounslow:barriers_london_rank -1.445e+00 9.061e-01
la19nmIslington:barriers_london_rank -7.852e-01 1.031e+00
la19nmKensington and Chelsea:barriers_london_rank 1.076e+00 1.007e+00
la19nmKingston upon Thames:barriers_london_rank -1.257e+00 9.816e-01
la19nmLambeth:barriers_london_rank -3.110e-01 9.491e-01
la19nmLewisham:barriers_london_rank 1.004e-01 9.045e-01
la19nmMerton:barriers_london_rank 4.934e-01 9.399e-01
la19nmNewham:barriers_london_rank 2.625e+00 2.260e+00
la19nmRedbridge:barriers_london_rank -1.116e+00 8.944e-01
la19nmRichmond upon Thames:barriers_london_rank -1.884e-01 1.104e+00
la19nmSouthwark:barriers_london_rank 5.627e-02 9.074e-01
la19nmSutton:barriers_london_rank -5.571e-02 9.399e-01
la19nmTower Hamlets:barriers_london_rank -2.394e+00 9.808e-01
la19nmWaltham Forest:barriers_london_rank -5.478e-01 9.237e-01
la19nmWandsworth:barriers_london_rank -2.874e-01 8.705e-01
la19nmWestminster:barriers_london_rank 3.494e+00 1.111e+00
t value Pr(>|t|)
(Intercept) 9.089 < 2e-16 ***
la19nmBarnet 4.929 8.57e-07 ***
la19nmBexley -2.210 0.027119 *
la19nmBrent 1.645 0.100112
la19nmBromley 0.749 0.454155
la19nmCamden -4.338 1.47e-05 ***
la19nmCity of London -0.159 0.873612
la19nmCroydon -2.277 0.022851 *
la19nmEaling 1.154 0.248528
la19nmEnfield 1.382 0.166890
la19nmGreenwich -1.491 0.135952
la19nmHackney -0.931 0.352136
la19nmHammersmith and Fulham -4.700 2.68e-06 ***
la19nmHaringey -0.229 0.818706
la19nmHarrow 3.150 0.001640 **
la19nmHavering -3.228 0.001254 **
la19nmHillingdon 0.384 0.701039
la19nmHounslow 2.393 0.016735 *
la19nmIslington -1.780 0.075114 .
la19nmKensington and Chelsea 5.787 7.61e-09 ***
la19nmKingston upon Thames 1.782 0.074808 .
la19nmLambeth -3.064 0.002197 **
la19nmLewisham -2.482 0.013106 *
la19nmMerton -3.019 0.002549 **
la19nmNewham 1.942 0.052165 .
la19nmRedbridge 2.013 0.044191 *
la19nmRichmond upon Thames 0.891 0.372776
la19nmSouthwark -3.552 0.000387 ***
la19nmSutton -3.169 0.001538 **
la19nmTower Hamlets -1.251 0.211119
la19nmWaltham Forest 1.141 0.253966
la19nmWandsworth 0.313 0.754638
la19nmWestminster -3.240 0.001203 **
livingEnv_london_rank -1.438 0.150471
barriers_london_rank 4.511 6.60e-06 ***
la19nmBarnet:livingEnv_london_rank 1.057 0.290593
la19nmBexley:livingEnv_london_rank 3.215 0.001312 **
la19nmBrent:livingEnv_london_rank 3.139 0.001705 **
la19nmBromley:livingEnv_london_rank 3.068 0.002169 **
la19nmCamden:livingEnv_london_rank -0.258 0.796425
la19nmCity of London:livingEnv_london_rank 0.364 0.716133
la19nmCroydon:livingEnv_london_rank 3.191 0.001425 **
la19nmEaling:livingEnv_london_rank 1.264 0.206349
la19nmEnfield:livingEnv_london_rank 3.354 0.000803 ***
la19nmGreenwich:livingEnv_london_rank 1.376 0.168758
la19nmHackney:livingEnv_london_rank -1.586 0.112711
la19nmHammersmith and Fulham:livingEnv_london_rank 0.946 0.344403
la19nmHaringey:livingEnv_london_rank -0.460 0.645286
la19nmHarrow:livingEnv_london_rank 2.005 0.044971 *
la19nmHavering:livingEnv_london_rank 3.307 0.000951 ***
la19nmHillingdon:livingEnv_london_rank 1.925 0.054317 .
la19nmHounslow:livingEnv_london_rank -0.223 0.823335
la19nmIslington:livingEnv_london_rank -2.273 0.023082 *
la19nmKensington and Chelsea:livingEnv_london_rank -6.137 9.08e-10 ***
la19nmKingston upon Thames:livingEnv_london_rank 1.459 0.144509
la19nmLambeth:livingEnv_london_rank -0.934 0.350602
la19nmLewisham:livingEnv_london_rank 1.087 0.277108
la19nmMerton:livingEnv_london_rank 3.082 0.002065 **
la19nmNewham:livingEnv_london_rank -0.027 0.978565
la19nmRedbridge:livingEnv_london_rank 1.396 0.162864
la19nmRichmond upon Thames:livingEnv_london_rank -0.014 0.988639
la19nmSouthwark:livingEnv_london_rank 1.683 0.092533 .
la19nmSutton:livingEnv_london_rank 3.370 0.000759 ***
la19nmTower Hamlets:livingEnv_london_rank 1.200 0.230337
la19nmWaltham Forest:livingEnv_london_rank -0.147 0.883300
la19nmWandsworth:livingEnv_london_rank -0.948 0.343019
la19nmWestminster:livingEnv_london_rank -4.578 4.80e-06 ***
la19nmBarnet:barriers_london_rank -1.558 0.119396
la19nmBexley:barriers_london_rank -0.510 0.609807
la19nmBrent:barriers_london_rank -0.787 0.431493
la19nmBromley:barriers_london_rank -1.431 0.152500
la19nmCamden:barriers_london_rank 2.555 0.010647 *
la19nmCity of London:barriers_london_rank 0.328 0.742734
la19nmCroydon:barriers_london_rank -0.721 0.470718
la19nmEaling:barriers_london_rank -0.676 0.499110
la19nmEnfield:barriers_london_rank -0.768 0.442499
la19nmGreenwich:barriers_london_rank -0.905 0.365429
la19nmHackney:barriers_london_rank 0.323 0.746543
la19nmHammersmith and Fulham:barriers_london_rank 1.530 0.126056
la19nmHaringey:barriers_london_rank 0.504 0.614335
la19nmHarrow:barriers_london_rank -1.535 0.124794
la19nmHavering:barriers_london_rank -0.417 0.676607
la19nmHillingdon:barriers_london_rank -1.834 0.066734 .
la19nmHounslow:barriers_london_rank -1.595 0.110827
la19nmIslington:barriers_london_rank -0.762 0.446220
la19nmKensington and Chelsea:barriers_london_rank 1.068 0.285397
la19nmKingston upon Thames:barriers_london_rank -1.280 0.200487
la19nmLambeth:barriers_london_rank -0.328 0.743140
la19nmLewisham:barriers_london_rank 0.111 0.911614
la19nmMerton:barriers_london_rank 0.525 0.599625
la19nmNewham:barriers_london_rank 1.161 0.245561
la19nmRedbridge:barriers_london_rank -1.247 0.212307
la19nmRichmond upon Thames:barriers_london_rank -0.171 0.864559
la19nmSouthwark:barriers_london_rank 0.062 0.950560
la19nmSutton:barriers_london_rank -0.059 0.952740
la19nmTower Hamlets:barriers_london_rank -2.440 0.014708 *
la19nmWaltham Forest:barriers_london_rank -0.593 0.553200
la19nmWandsworth:barriers_london_rank -0.330 0.741265
la19nmWestminster:barriers_london_rank 3.146 0.001667 **
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 5162 on 4736 degrees of freedom
Multiple R-squared: 0.5789, Adjusted R-squared: 0.5702
F-statistic: 66.45 on 98 and 4736 DF, p-value: < 2.2e-16
Challenge 2
Using the gapminder
data to create a linear model
between a categorical and a continuous variable .
Discuss your question and your findings.
Broom
The ‘broom’ package offers an alternative way of presenting the output of statistical analysis. It centers around three S3 methods, each of which take common objects produced by R statistical functions (lm, t.test, nls, etc) and convert them into a tibble.
These are:
- tidy: constructs a tibble that summarizes the model’s statistical findings. This includes coefficients and p-values for each term in a regression, per-cluster information in clustering applications, or per-test information for multtest functions.
- augment: add columns to the original data that was modeled. This includes predictions, residuals, and cluster assignments.
- glance: construct a concise one-row summary of the model. This typically contains values such as R^2, adjusted R^2, and residual standard error that are computed once for the entire model.
Let’s revisit our initial linear model:
R
reg_LivEnv_health <- lm(health_london_rank ~ livingEnv_london_rank, data = lon_dims_imd_2019)
summary(reg_LivEnv_health)
OUTPUT
Call:
lm(formula = health_london_rank ~ livingEnv_london_rank, data = lon_dims_imd_2019)
Residuals:
Min 1Q Median 3Q Max
-21549.6 -5948.2 609.1 6239.2 15792.2
Coefficients:
Estimate Std. Error t value Pr(>|t|)
(Intercept) 1.692e+04 2.274e+02 74.40 <2e-16 ***
livingEnv_london_rank 3.430e-01 1.915e-02 17.91 <2e-16 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Residual standard error: 7625 on 4833 degrees of freedom
Multiple R-squared: 0.06225, Adjusted R-squared: 0.06205
F-statistic: 320.8 on 1 and 4833 DF, p-value: < 2.2e-16
There is a lot of useful information, but it not available in a way so that you can combine it with other models or do further analysis. We can convert this to tabular data using the ‘tidy’ function.
R
tidy(reg_LivEnv_health)
OUTPUT
# A tibble: 2 × 5
term estimate std.error statistic p.value
<chr> <dbl> <dbl> <dbl> <dbl>
1 (Intercept) 16916. 227. 74.4 0
2 livingEnv_london_rank 0.343 0.0192 17.9 1.63e-69
The row names have been moved into a column called term, and the column names are simple and consistent (and can be accessed using $).
Information about the model can be explored with ‘augment’. The function augments the original data with information from the model, such as the fitted values and residuals for each of the original points in the regression.
R
augment(reg_LivEnv_health)
OUTPUT
# A tibble: 4,835 × 8
health_london_rank livingEnv_london_rank .fitted .resid .hat .sigma
<int> <int> <dbl> <dbl> <dbl> <dbl>
1 32113 7789 19588. 12525. 0.000250 7624.
2 29705 13070 21400. 8305. 0.000252 7625.
3 17600 4092 18320. -720. 0.000458 7626.
4 17907 9397 20140. -2233. 0.000213 7626.
5 21581 10629 20562. 1019. 0.000207 7626.
6 16414 11162 20745. -4331. 0.000210 7626.
7 12334 8672 19891. -7557. 0.000226 7625.
8 9661 9611 20213. -10552. 0.000211 7625.
9 16050 2269 17694. -1644. 0.000624 7626.
10 18178 4309 18394. -216. 0.000441 7626.
# ℹ 4,825 more rows
# ℹ 2 more variables: .cooksd <dbl>, .std.resid <dbl>
Some of the data presented by ‘augment’ will be discussed in the episode Linear Regression Diagnostics.
Summary statistics are computed for the entire regression, such as R^2 and the F-statistic can be accessed with the ‘glance’ function:
R
glance(reg_LivEnv_health)
OUTPUT
# A tibble: 1 × 12
r.squared adj.r.squared sigma statistic p.value df logLik AIC BIC
<dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
1 0.0622 0.0621 7625. 321. 1.63e-69 1 -50081. 100167. 100187.
# ℹ 3 more variables: deviance <dbl>, df.residual <int>, nobs <int>
Generalised linear models
We can also use the ‘broom’ functions to present data from Generalised linear and non-linear models. For example, if we wanted to explore the Income Rank in relation to whether or not an area was within the City of London.
R
# add a variable to indicate whether or not an area is within the City of London
lon_dims_imd_2019 <- lon_dims_imd_2019 %>% mutate(city = la19nm == "City of London")
# create a Generalised Linear Model
glmlondims <- glm(city ~ Income_london_rank, lon_dims_imd_2019, family = "binomial")
summary(glmlondims)
OUTPUT
Call:
glm(formula = city ~ Income_london_rank, family = "binomial",
data = lon_dims_imd_2019)
Coefficients:
Estimate Std. Error z value Pr(>|z|)
(Intercept) -7.868e+00 1.004e+00 -7.835 4.68e-15 ***
Income_london_rank 6.888e-05 4.635e-05 1.486 0.137
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
(Dispersion parameter for binomial family taken to be 1)
Null deviance: 92.295 on 4834 degrees of freedom
Residual deviance: 90.062 on 4833 degrees of freedom
AIC: 94.062
Number of Fisher Scoring iterations: 10
Use of ‘tidy’:
R
tidy(glmlondims)
OUTPUT
# A tibble: 2 × 5
term estimate std.error statistic p.value
<chr> <dbl> <dbl> <dbl> <dbl>
1 (Intercept) -7.87 1.00 -7.84 4.68e-15
2 Income_london_rank 0.0000689 0.0000463 1.49 1.37e- 1
Use of ‘augment’:
R
augment(glmlondims)
OUTPUT
# A tibble: 4,835 × 8
city Income_london_rank .fitted .resid .hat .sigma .cooksd .std.resid
<lgl> <int> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
1 TRUE 32831 -5.61 3.35 0.00194 0.128 2.65e-1 3.35
2 TRUE 29901 -5.81 3.41 0.00115 0.127 1.93e-1 3.41
3 TRUE 18510 -6.59 3.63 0.000233 0.126 8.51e-2 3.63
4 TRUE 6029 -7.45 3.86 0.000332 0.125 2.87e-1 3.86
5 FALSE 14023 -6.90 -0.0448 0.000239 0.137 1.20e-7 -0.0448
6 FALSE 6261 -7.44 -0.0343 0.000330 0.137 9.72e-8 -0.0343
7 FALSE 3382 -7.64 -0.0311 0.000360 0.137 8.70e-8 -0.0311
8 FALSE 7506 -7.35 -0.0358 0.000315 0.137 1.01e-7 -0.0358
9 FALSE 8902 -7.25 -0.0376 0.000298 0.137 1.05e-7 -0.0376
10 FALSE 9033 -7.25 -0.0378 0.000296 0.137 1.06e-7 -0.0378
# ℹ 4,825 more rows
Use of ‘glance’:
R
glance(glmlondims)
OUTPUT
# A tibble: 1 × 8
null.deviance df.null logLik AIC BIC deviance df.residual nobs
<dbl> <int> <dbl> <dbl> <dbl> <dbl> <int> <int>
1 92.3 4834 -45.0 94.1 107. 90.1 4833 4835
You will notice that the statistics computed by ‘glance’ are different for glm objects than for lm (e.g. deviance rather than R^2).
Hypothesis testing
The tidy function can also be applied a range of hypotheses tests, such as built-in functions like t.test, cor.test, and wilcox.test.
t-test
R
tt <- t.test(Income_london_rank ~ city, lon_dims_imd_2019)
tidy(tt)
OUTPUT
# A tibble: 1 × 10
estimate estimate1 estimate2 statistic p.value parameter conf.low conf.high
<dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
1 -5344. 14456. 19800. -1.24 0.270 5.01 -16407. 5719.
# ℹ 2 more variables: method <chr>, alternative <chr>
Some cases might have fewer columns (for example, no confidence interval).
Wilcox test:
R
wt <- wilcox.test(Income_london_rank ~ city, lon_dims_imd_2019)
tidy(wt)
OUTPUT
# A tibble: 1 × 4
statistic p.value method alternative
<dbl> <dbl> <chr> <chr>
1 9836. 0.174 Wilcoxon rank sum test with continuity correcti… two.sided
Since the ‘tidy’ output is already only one row, glance returns the same output:
R
# t-test
glance(tt)
OUTPUT
# A tibble: 1 × 10
estimate estimate1 estimate2 statistic p.value parameter conf.low conf.high
<dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
1 -5344. 14456. 19800. -1.24 0.270 5.01 -16407. 5719.
# ℹ 2 more variables: method <chr>, alternative <chr>
R
# Wilcox test
glance(wt)
OUTPUT
# A tibble: 1 × 4
statistic p.value method alternative
<dbl> <dbl> <chr> <chr>
1 9836. 0.174 Wilcoxon rank sum test with continuity correcti… two.sided
The chisq.test enables use to investigate whether changes in one categorical variable are related to changes in another categorical variable.
The ‘augment’ method is defined only for chi-squared tests, since there is no meaningful sense, for other tests, in which a hypothesis test produces output about each initial data point.
R
# convert IDAOP_london_decile to a factor so it is not interprested as continuous data
lon_dims_imd_2019$IDAOP_london_decile <- factor(lon_dims_imd_2019$IDAOP_london_decile)
# xtabs creates a frequency table of IMD deciles within London borooughs
chit <- chisq.test(xtabs(~ la19nm + IDAOP_london_decile, data = lon_dims_imd_2019))
WARNING
Warning in chisq.test(xtabs(~la19nm + IDAOP_london_decile, data =
lon_dims_imd_2019)): Chi-squared approximation may be incorrect
R
tidy(chit)
OUTPUT
# A tibble: 1 × 4
statistic p.value parameter method
<dbl> <dbl> <int> <chr>
1 2841. 0 288 Pearson's Chi-squared test
R
augment(chit)
OUTPUT
# A tibble: 330 × 9
la19nm IDAOP_london_decile .observed .prop .row.prop .col.prop .expected
<fct> <fct> <int> <dbl> <dbl> <dbl> <dbl>
1 Barking … 1 6 1.24e-3 0.0545 0.0124 11.0
2 Barnet 1 6 1.24e-3 0.0284 0.0124 21.1
3 Bexley 1 0 0 0 0 14.6
4 Brent 1 12 2.48e-3 0.0694 0.0248 17.3
5 Bromley 1 2 4.14e-4 0.0102 0.00414 19.7
6 Camden 1 12 2.48e-3 0.0902 0.0248 13.3
7 City of … 1 0 0 0 0 0.599
8 Croydon 1 8 1.65e-3 0.0364 0.0166 22.0
9 Ealing 1 11 2.28e-3 0.0561 0.0228 19.6
10 Enfield 1 9 1.86e-3 0.0492 0.0186 18.3
# ℹ 320 more rows
# ℹ 2 more variables: .resid <dbl>, .std.resid <dbl>
There are a number of underlying assumptions of a Chi-Square test, these are:
Independence: The Chi-Square test assumes that the observations in the data are independent of each other. This means that the outcome of one observation should not influence the outcome of another.
Random sampling: The data should be obtained through random sampling to ensure that it is representative of the population from which it was drawn.
Expected frequency: The Chi-Square test assumes that the expected frequency count for each cell in the contingency table should be at least 5. If this assumption is not met, the test results may not be reliable.
As we have received a warning about the reliability of our test, it is likely that one of these assumptions has not been met, and that this is not a suitable test for this data.
Challenge 3
Use broom to amend the display of your model outputs.
Which function(s) did you use and why?
Conventions
There are some conventions that enable consistency across the broom functions, these are: * The output of the tidy, augment and glance functions is always a tibble.
- The output never has rownames. This ensures that you can combine it with other tidy outputs without fear of losing information (since rownames in R cannot contain duplicates).
- Some column names are kept consistent, so that they can be combined across different models and so that you know what to expect (in contrast to asking “is it pval or PValue?” every time).