Chapter 7 Regression Assumptions

7.1 Seminar – in class exercises

7.1.1 Required Packages

Today, we need three new packages: lmtest, sandwich and gridExtra. We will also load tidyverse and texreg.

install.packages("lmtest")
install.packages("sandwich")
install.packages("gridExtra")

Now load the libraries.

library(lmtest)
library(sandwich)
library(texreg)
library(tidyverse)
library(gridExtra)

7.1.2 Omitted Variable Bias

We have already dealt with OVB at length, but it is worth going over another (silly) example here for revision purposes (we talked about this example in class as well).

7.1.2.1 Ice Cream and Shark Attacks

Does ice cream consumption increase the likelihood of being attacked by a shark? Let’s explore a dataset of shark attacks, ice cream sales and monthly temperatures collected over 7 years to find out.

shark_attacks <- read.csv("https://raw.githubusercontent.com/QMUL-SPIR/Public_files/master/datasets/shark_attacks.csv")
head(shark_attacks)
  Year Month SharkAttacks Temperature IceCreamSales
1 2008     1           25        11.9            76
2 2008     2           28        15.2            79
3 2008     3           32        17.2            91
4 2008     4           35        18.5            95
5 2008     5           38        19.4           103
6 2008     6           41        22.1           108

Run a linear model to see the effects of ice cream consumption on shark attacks.

model1 <- lm(SharkAttacks ~ IceCreamSales, data = shark_attacks)

screenreg(model1)

========================
               Model 1  
------------------------
(Intercept)     4.35    
               (5.24)   
IceCreamSales   0.34 ***
               (0.06)   
------------------------
R^2             0.29    
Adj. R^2        0.28    
Num. obs.      84       
========================
*** p < 0.001; ** p < 0.01; * p < 0.05

Let’s check pairwise correlation between ice cream consumption, average monthly temperatures and shark attacks.

shark_attacks <-
  select(shark_attacks, # dataset
         SharkAttacks, # relevant vars
         Temperature, 
         IceCreamSales)

cor(shark_attacks) # inspect correlations
              SharkAttacks Temperature IceCreamSales
SharkAttacks     1.0000000   0.7169660     0.5343576
Temperature      0.7169660   1.0000000     0.5957694
IceCreamSales    0.5343576   0.5957694     1.0000000

These two steps could have been combined with the ‘pipe’ – to see how, take a look at Going further with the tidyverse: dplyr, magrittr and ggplot2.

It looks like there’s high correlation between average monthly temperatures and shark attacks, and also between ice cream consumption and monthly temperatures, so let’s add the Temperature variable to our model.

model2 <- lm(SharkAttacks ~ IceCreamSales + Temperature, data = shark_attacks)

Now compare it to the first model.

screenreg(list(model1, model2))

===================================
               Model 1    Model 2  
-----------------------------------
(Intercept)     4.35       0.57    
               (5.24)     (4.31)   
IceCreamSales   0.34 ***   0.10    
               (0.06)     (0.06)   
Temperature                1.29 ***
                          (0.20)   
-----------------------------------
R^2             0.29       0.53    
Adj. R^2        0.28       0.52    
Num. obs.      84         84       
===================================
*** p < 0.001; ** p < 0.01; * p < 0.05

The second model shows that once we control for monthly temperatures, ice cream consumption has no effect on the number of shark attacks.

What was the bias in the coefficient of IceCreamSales in the model where we did not control for Temperature?

# coefficient in new model minus coefficient in old model
0.10 -  0.34
[1] -0.24

We overestimated the effect by 0.24, more than two thirds of the effect. We know that the real effect of selling ice cream on shark attacks is exactly 0 (unless for some reason sharks really like icecream). The two phenomena are unrelated. Our estimate, although insignificant, is still an over-estimate. The fact that it is insignificant, though, tells us that it is essentially indistinguishable from zero.

What is likely to be happening here? Higher temperatures lead people to eat more ice cream. Higher temperatures also lead people to swim in the sea, and since sharks can (pretty much) only attack you if you are in the sea, this therefore means there are more shark attacks when it is hotter. Ice cream consumption, however, has no effect on shark attacks. But, because they share a common cause (temperature), a regression model like model1 makes it look like such an effect does exist, because of omitted variable bias.

7.1.3 Detecting non-linearity

For this part of the session, we will be using data on the results of the German Bundestag elections in 2017. You can load this data directly from GitHub.

german_results <- 
  read.csv("https://raw.githubusercontent.com/QMUL-SPIR/Public_files/master/datasets/german_election_results.csv",
           fileEncoding = "utf-8") 

Note, the data contains German characters. To get properly displayed, we load the data specifying its character encoding utf-8.

We should again quickly see the variables that are included in the data using the head() function (or any other way you prefer):

head(german_results)
                           district      CDU      SPD      AfD Die.Linke
1             Flensburg – Schleswig 39.99930 28.01454 6.214686  7.129180
2 Nordfriesland – Dithmarschen Nord 45.14833 25.15210 5.881031  5.150071
3      Steinburg – Dithmarschen Süd 41.87862 26.14625 7.642704  5.481995
4             Rendsburg-Eckernförde 42.68043 28.87215 6.826306  5.172259
5                              Kiel 30.50011 31.05149 6.136603  7.376207
6                 Plön – Neumünster 40.69998 28.91278 7.834377  5.371334
      Grune       FDP unemploymentrate migrantfraction catholicfraction
1 10.512869  6.544773             7.16            10.0              5.1
2  9.453158  8.051428             7.05             8.0              5.2
3  6.716686 11.033518             6.30             9.2              4.2
4  8.954402  6.455395             5.18             8.5              4.9
5 14.403295  7.462318             9.65            18.0              7.3
6  8.993862  7.187579             6.72            10.8              5.6

For this problem, we are going to be interested in modelling the share of the vote won by the SPD in each of the 299 electoral districts. This variable is named SPD in the german_results data.frame. Let’s build a linear regression model with two predictors: migrantfraction and catholicfraction, both of which are measured on a range from 0 to 100 percent.

# model on SPD
spd_linear_model <- lm(SPD ~ migrantfraction + catholicfraction , data = german_results)

# show model
screenreg(spd_linear_model)

============================
                  Model 1   
----------------------------
(Intercept)        19.47 ***
                   (1.07)   
migrantfraction     0.38 ***
                   (0.05)   
catholicfraction   -0.06 ** 
                   (0.02)   
----------------------------
R^2                 0.16    
Adj. R^2            0.16    
Num. obs.         299       
============================
*** p < 0.001; ** p < 0.01; * p < 0.05

The model suggests that the fraction of migrants in a district is positively related to the popularity of the SPD in that district, and that there is a negative relationship between the fraction of catholics in a district and the SPD vote share. More precisely, the model tells us that for each additional percentage point of migrants, the SPD’s vote share increases by 0.38 points, holding constant the fraction of Catholics. Conversely, for each additional percentage point of Catholics, the SPD’s share of the vote increases by -0.06 points, holding constant the fraction of migrants.

This might be useful for us if we were concerned that the relationship between higher proportions of migrants and SPD vote share is actually just accounted for by the fact that migrants are less or more likely to be Catholic. We do our best to rule out that possibility by including the catholicfraction variable in our analysis.

As discussed in lecture, it is often useful to examine plots of the model residuals to determine whether there is evidence of a non-linear relationship between our dependent variable and our independent variables. Recall that the sample residuals are the difference between the observed values of Y and the predicted values of Y given our model: \(u_i = Y_i - \hat{Y}_i\). We can extract the residuals directly from our model and add them to our dataset.

# get the residuals
german_results$residuals <- spd_linear_model$residuals

If you wanted to estimate the residuals from the fitted values manually, you could do so:

german_results$residuals2 <- german_results$SPD - spd_linear_model$fitted.values

Assess whether both methods lead to the same result. Hint: There is rounding error, so both methods may not produce exactly the same results. Confirm that the results are equivalent nonetheless.

# correlation coefficient
cor(german_results$residuals, german_results$residuals2)
[1] 1

To assess whether there is evidence for non-linearity, it can be helpful to plot the residuals against the continuous explanatory variables in our model. Remember, the assumptions behind OLS models suggest that the residuals should be randomly distributed around zero for all values of X: \(E(u_i|X_i) = 0\). So, any pattern that deviates from this randomness suggests that we may have misspecified our model.

Let’s plot the residuals from spd_linear_model against the two independent variables in our data. We use the grid.arrange() function from gridExtra to put the two plots side by side.

p1 <- ggplot(data = german_results, 
             mapping = aes(x = catholicfraction, 
                           y = residuals)) + 
  geom_point() + 
  labs(x = "Catholic Fraction", 
       y = "Residuals") + 
  geom_hline(yintercept = 0) + 
  theme_bw()

p2 <- ggplot(data = german_results, 
             mapping = aes(x = migrantfraction, 
                           y = residuals)) + 
  geom_point() + 
  labs(x = "Migrant Fraction", 
       y = "Residuals") + 
  geom_hline(yintercept = 0) + 
  theme_bw()

grid.arrange(p1, p2,  # use grid.arrange to plot them together
             ncol = 2) # makes two 'columns' so they are side by side

What can we conclude from these plots? In the left-hand plot, the residuals seem to be distributed fairly randomly around zero for all values of catholicfraction: there does not appear to be evidence of non-linearity for this variable.

In the right-hand plot, however, it seems that the residuals are mostly negative for both low and high levels of the migrantfraction variable, but evenly spread around zero for the more moderate range of this variable. This pattern of residuals does seem to deviate from the OLS assumptions to a certain degree, and therefore suggests that we might want to try to account for this possible non-linearity in our model.

One way of doing this is to include a polynomial term (we will use \(X^2\) here) in our model. Let’s do that now, and then screenreg() the results.

spd_non_linear_model <- lm(SPD ~ poly(migrantfraction,2) + catholicfraction , data = german_results)
screenreg(spd_non_linear_model)

=====================================
                           Model 1   
-------------------------------------
(Intercept)                 28.57 ***
                            (0.75)   
poly(migrantfraction, 2)1   66.31 ***
                            (7.28)   
poly(migrantfraction, 2)2  -58.19 ***
                            (7.71)   
catholicfraction            -0.13 ***
                            (0.02)   
-------------------------------------
R^2                          0.30    
Adj. R^2                     0.29    
Num. obs.                  299       
=====================================
*** p < 0.001; ** p < 0.01; * p < 0.05

The coefficient on the squared term (poly(migrantfraction, 2)2) is significant, which indicates that there is evidence of a non-linear relationship between migrantfraction and SPD vote share in our data. This suggests that our interpretation of the residual plot above was correct, and that including the squared term is an important modification to our model. (You might also note that the Adjusted \(R^2\) has nearly doubled with the inclusion of \(X^2\), again suggesting that the non-linear adjustment is important.)

We can now repeat the same residual-plotting exersise with the residuals from our new model, to see if we are closer to meeting the OLS assumption that the error term is randomly distributed around zero for all values of X.

german_results$residuals_model_two <- residuals(spd_non_linear_model)

p3 <- ggplot(data = german_results, 
             mapping = aes(x = catholicfraction, 
                           y = residuals_model_two)) + 
  geom_point() + 
  labs(x = "Catholic Fraction", 
       y = "Residuals") + 
  geom_hline(yintercept = 0) + 
  theme_bw()

p4 <- ggplot(data = german_results, 
             mapping = aes(x = migrantfraction, 
                           y = residuals_model_two)) + 
  geom_point() + 
  labs(x = "Migrant Fraction", 
       y = "Residuals") + 
  geom_hline(yintercept = 0) + 
  theme_bw()

grid.arrange(p3, p4, ncol = 2)

The right-hand plot now looks much better: the residuals seem randomly distributed around zero for all values of the migrantfraction variable, and there is no clearly distinguishable pattern anymore.

These side-by-side plots are similar to ‘facets’ in ggplot2. To learn about those, see Going further with the tidyverse: dplyr, magrittr and ggplot2.

You can also inspect how the model does overall. Repeat the previous exercise but plot the fitted values on the x-axis. What can we conclude?

german_results$fitted.values <- spd_linear_model$fitted.values

p5 <- ggplot(data = german_results, 
             mapping = aes(y = residuals_model_two, 
                           x = fitted.values)) + 
  geom_point() +
  geom_hline(yintercept = 0) + 
  labs(x = "Fitted values", 
       y = "Residuals") + 
  theme_bw()

p5

Overall, it looks like conditional mean independence is satisfied.

7.1.4 Heteroskedasticity

In order to understand heteroskedasticity, let’s start by loading a sample of the U.S. Current Population Survey (CPS) from 2013. The dataset contains 2989 observations of full-time workers with variables including age, gender, years of education and income reported in hourly earnings.

cps <- read.csv("https://raw.githubusercontent.com/QMUL-SPIR/Public_files/master/datasets/cps2013.csv")

head(cps)
  age gender education    income
1  30   Male        13 12.019231
2  30   Male        12  9.857142
3  30   Male        12  8.241758
4  29 Female        12 10.096154
5  30   Male        14 24.038462
6  30   Male        16 23.668638

We plot income by years of education.

p6 <- ggplot(data = cps, 
             mapping = aes(x = education, 
                           y = income)) + 
  geom_point(alpha = 0.3) + # more transparency in points
  labs(x = "Level of education",
       y = "Income") + 
  theme_bw()
  
p6

The dots in this plot largely overlap because education is only collected for full years. It’s therefore a bit hard to see the amount of observations that have 12 years of education and income 20. You can make it easier to see the amount of points by adding sine random noise to the education variable. So for example, for everyone with twelve years of eductation, we add or subtract a small amount at random. We will still be able to tell the education categories apart. We use this geom_jitter() function to do this. We choose a small number of noise to add such that we can see the number of observations better and education categories are still visible. You have to play around to find a good number.

p7 <- ggplot(data = cps, 
             mapping = aes(x = education, 
                           y = income)) + 
  geom_jitter(alpha = 0.3, # geom_jitter() plots points but with random noise to spread them out
              width = 0.25) + 
  labs(x = "Level of education",
       y = "Income") + 
  theme_bw()
  
p7

We can see that the range of values for income have a larger spread as the level of education increases. Intuitively this makes sense as people with more education tend to have more opportunities and are employed in a wider range of professions. But how does this larger spread affect the regression model?

Let’s run linear regression and take a look at the results to find out the answer.

model1 <- lm(income ~ education, data = cps)
summary(model1)

Call:
lm(formula = income ~ education, data = cps)

Residuals:
    Min      1Q  Median      3Q     Max 
-23.035  -6.086  -1.573   3.900  60.446 

Coefficients:
            Estimate Std. Error t value Pr(>|t|)    
(Intercept) -5.37508    1.03258  -5.206 2.07e-07 ***
education    1.75638    0.07397  23.745  < 2e-16 ***
---
Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1

Residual standard error: 9.496 on 2987 degrees of freedom
Multiple R-squared:  0.1588,    Adjusted R-squared:  0.1585 
F-statistic: 563.8 on 1 and 2987 DF,  p-value: < 2.2e-16

The model tells us the for each additional year of education, the income increases by 1.756.

Now let’s plot the fitted model on our scatter plot of observations.

p8 <- p7 + geom_smooth(method = "lm")

p8
`geom_smooth()` using formula = 'y ~ x'

Looking at the fitted model, we can see that the errors (i.e. the differences between the predicted value and the observed value) increase as the level of education increases. This is what is known as heteroskedasticity or heteroskedastic errors. In plain language it simply means that the variability of the error term is not constant across the entire range of our observations.

Another way to visualize this is by plotting residuals against the fitted values.

new_data <- data.frame(residuals = model1$residuals,
                       fitted.values = model1$fitted.values)

p9 <- ggplot(data = new_data, 
             mapping = aes(x = fitted.values, 
                           y = residuals)) + 
  geom_jitter(alpha = 0.3, 
              width = 0.5) +
  labs(x = "Predicted income", 
       y = "Residuals (difference b/w observed and fitted value)",
       title = "Residuals vs. Fitted") +
  theme_bw()

p9

Again, we can see that residuals are not constant and instead increase as the fitted values increase. In addition to visual inspection, we can also test for heteroskedasticity using the Breusch-Pagan Test from the lmtest package we loaded at the beginning.

bptest(model1)

    studentized Breusch-Pagan test

data:  model1
BP = 76.679, df = 1, p-value < 2.2e-16

The null hypothesis for the Breusch-Pagan test is that the variance of the error term is constant, or in other words, the errors are homoskedestic. By looking at the p-value from Breusch-Pagan test we can determine whether we have heteroskedastic errors or not.

The p-value tells us that we can reject the null hypothesis of homoskedestic errors. Once we’ve determined that we’re dealing with heteroskedastic errors, we can correct them using the coeftest() function from the lmtest package.

Here is a list of arguments for the coeftest() function:

coeftest(model, vcov)
Argument Description
model This is the estimated model obtained from lm().
vcov Covariance matrix. The simplest way to obtain heteroskedasticity-consistent covariance matrix is by using the vcovHC() function from the sandwich package. The result from vcovHC() can be directly passed to coeftest() as the second argument.
screenreg(coeftest(model1, vcov = vcovHC(model1)))

======================
             Model 1  
----------------------
(Intercept)  -5.38 ***
             (1.05)   
education     1.76 ***
             (0.08)   
======================
*** p < 0.001; ** p < 0.01; * p < 0.05

After correcting for heteroskedasticity we can see that the standard error for the independent variable education have increased from 0.074 to 0.08. Even though this increase might seem very small, remember that it is relative to the scale of the independent variable. Since the standard error is the measure of precision for the esitmated coefficients, an increase in standard error means that our original estiamte wasn’t as good as we thought it was before we corrected for heteroskedasticity.

Now that we can get heteroskedastic corrected standard errors, how would you present them in a publication (or in your dissertation or the final exam)? Fortunately, all texreg functions such as screenreg() or htmlreg() allow us to easily override the standard errors and p-value in the resulting output.

We first need to save the result from coeftest() to an object and then override the output from screenreg() by extracting the two columns of interest.

The corrected standard errors are in column 2 of the object returned by coeftest() and the associated p-values are in column 4:

corrected_errors <- coeftest(model1, vcov = vcovHC(model1)) 

screenreg(model1, 
          override.se = corrected_errors[, 2], 
          override.pval = corrected_errors[, 4])

========================
             Model 1    
------------------------
(Intercept)    -5.38 ***
               (1.05)   
education       1.76 ***
               (0.08)   
------------------------
R^2             0.16    
Adj. R^2        0.16    
Num. obs.    2989       
========================
*** p < 0.001; ** p < 0.01; * p < 0.05

7.1.5 Homework exercises

There are lots of exercises this week, on three different datasets, to give you plenty of opportunity to practise the things we are doing in the seminar.

Does Indiscriminate Violence Incite Insurgent Attacks?

chechen <- read.csv("https://raw.githubusercontent.com/QMUL-SPIR/Public_files/master/datasets/chechen.csv")

head(chechen)
  pair     date year      village          district vilno disno treat poverty
1    1 6/1/2000 2000   Elistanzhi         Vedenskiy    36    15     0       3
2    1 6/1/2000 2000 Malye Shuani Nozhai-Yurtovskiy    79     9     1       3
3    2 6/5/2000 2000     Belgatoy        Shalinskiy    15    10     1       2
4    2 6/5/2000 2000 Oktya'brskoe         Vedenskiy    98    15     0       3
5    3 6/5/2000 2000   Chiri-Yurt        Shalinskiy    24    10     0       2
6    3 6/5/2000 2000    Gansolchu Nozhai-Yurtovskiy    40     9     1       3
  pop2000 lpop2000 pret pret.1 diff groznyy elevation    lelev tariq     nn
1    1845 7.520235    4      4   -1       0       702 6.553934     0 2.8470
2     323 5.777652    0      0    1       0       670 6.507277     0 1.4816
3    3395 8.130059    1      1   -1       0       809 6.695799     0 4.8000
4    2724 7.909857    0      0    0       0       667 6.502790     0 1.4816
5   11316 9.333973    4      4    1       0       313 5.746203     0 1.1930
6     823 6.712956    0      0    0       0       738 6.603944     0 1.8000
        lnn reb garrison vildays deaths deathsd wounded woundedd fail swept
1 1.0462660   1        1      28     NA      NA      NA       NA    1     0
2 0.3931226   1        0       5      0       0       0        0    1     0
3 1.5686160   1        0      90     34       1       0        0    0     0
4 0.3931226   1        0      90     NA      NA      NA       NA    0     0
5 0.1764711   1        1      25     NA      NA      NA       NA    1     0
6 0.5877866   1        0      90      0       0       0        0    0     0
  prop propd    lprop history historyd lhistory change decrease increase
1   NA    NA       NA      NA       NA       NA     -1        1        0
2    0     0 0.000000       0        0        0      1        0        1
3    0     0 0.000000       0        0        0     -1        1        0
4   NA    NA       NA      NA       NA       NA      0        0        0
5   NA    NA       NA      NA       NA       NA      1        0        1
6    2     1 1.098612       0        0        0      0        0        0
  abandon iso
1       0   4
2       3   6
3       2   0
4       0   5
5       0   4
6       0   5


Jason Lyall:
Does a state’s use of indiscriminate violence incite insurgent attacks? To date, most existing theories and empirical studies have concluded that such violence is highly counterproductive because it creates new grievances while forcing victims to seek security, if not safety, in rebel arms. This proposition is tested using Russian artillery fire in Chechnya (2000 to 2005) to estimate indiscriminate violence’s effect on subsequent patterns of insurgent attacks […].


Codebook for the chechen data:

Variable Description
vilno unique village number
disno unique distric number
treat Russian artillery shelling of cechan village
poverty estimated need (3 is highest)
pop2000 population in 2000
lpop2000 logged population in 2000
pret pre-treatment insurgent attacks
post post-treatment insurgent attacks
diff post minus pret
groznyy 1 if in groznyy
elevation in meters
lelev logged elevation
tariq Tariqa (1=Naq.)
nn distance to nearest neighbour (km)
lnn logged distance to nearest neighbour
reb rebel (1 = Basayev)
garrison (1 = Russian base present)
vildays days to first insurgent attack after treatment
deaths estimated number of individuals killed during Russian artillery shelling
deathsd 1 indicates that individuals were killed during shelling
wounded estimated number of individuals wounded during Russian artillery shelling
woundedd 1 indicates that individuals were wounded during Russian artillery shelling
fail 1 if insurgent attack is observed in 90 days after artillery shelling
swept 1 indicates village was swept in 90 days prior to artillery shelling
prop estimated number of structures destroyed during shelling
propd 1 indicates that property was destroyed during shelling
lprop logged number of estimated structures destroyed or damaged during shelling
history number of times village had been shelled in the past
historyd 1 indicates that village had been shelled in the past
lhistory logged number of times village had been shelled
change 1 if DD is increase: 0 if no change, -1 if DD decreases
decrease 1 if decrease recorded, 0 if not
increase 1 if increase in DD recorded, 0 if not
abandon number of neighbours within 5km^2 radius


The dataset comes from an article in the Journal of Conflict Resolution. Use the codebook above for the tasks and if anything is unclear, refer to the article that we have linked to above.

  1. Run a bivariate regression model with diff as the dependent variable and treat as the predictor.
  2. Control for additional variables. Discuss your choice of variables and refer to omitted variable bias.
  3. Discuss the improvement in model fit and use the appropriate test statistic.
  4. Check for non-linearity and if there is solve the problem.
  5. Check for heteroskedasticity and if there is solve the problem.
  6. Thoroughly compare the models, and interpret the outcome in substantial as well statistical terms - do not forget to talk about model fit. What are the implications of this?


Ordinary Economic Voting Behavior in the Extraordinary Election of Adolf Hitler

nazi <- read.csv("https://raw.githubusercontent.com/QMUL-SPIR/Public_files/master/datasets/who_voted_nazi_in_1932.csv")

head(nazi)
  shareself shareblue sharewhite sharedomestic shareunemployed nvoter nazivote
1  14.52451  46.67877    7.51894      23.72128         7.55650  20724     8915
2  17.76995  37.84700    8.86076      27.10842         8.41387  61024    20388
3  21.00499  28.78295    6.16480      37.18949         6.85777  31741     9670
4  14.98590  45.92575    9.11449      21.09423         8.87963  67591    26634
5  11.46592  27.82162   29.76835       7.48931        23.45480 226176   100041
6  21.45603  30.40565    5.73774      35.49705         6.90353  30835    15255
  sharenazis
1   43.01776
2   33.40981
3   30.46533
4   39.40465
5   44.23148
6   49.47300


The goal of analysis is to investigate which types of voters (based on their occupation category) cast ballots for the Nazis. One hypothesis says that the Nazis received much support from blue-collar workers. Since the data do not directly tell us how many blue-collar workers voted for the Nazis, we must infer this information using a statistical analysis […].

The data contains aggregate voting behaviour from 681 voting districts.


Codebook:

Variable Description
sharenazis Percent of the vote Nazis received in the district
nazivote Number of Nazi votes
nvoter Total number of eligible voters
shareblue Percent of blue-collar potential voters
sharewhite Percent of white-collar potential voters
shareself Percent of self-employed potential voters
sharedomestic Percent of domestically employed potential voters
shareunemployed Percent of unemployed potential voters
  1. Run a bivariate model to test whether blue-collar voters voted for the Nazis.
  2. Control for additional variables. Discuss your choice of variables and refer to omitted variable bias.
  3. Can you fit a model using shareblue, sharewhite, shareself, sharedomestic, and shareunemployed as predictors? Why or why not?
  4. Check for heteroskedasticity and if there is solve the problem.
  5. Interpret your models.


Finally we load the Massachusetts Test Scores dataset MA_Schools.csv.

ma <- read.csv("https://raw.githubusercontent.com/QMUL-SPIR/Public_files/master/datasets/MA_Schools.csv")

head(ma)
  district municipality expreg expspecial expbil expocc exptot scratio special
1        1     Abington   4201    7375.69      0      0   4646    16.6    14.6
2        2        Acton   4129    8573.99      0      0   4930     5.7    17.4
3        3     Acushnet   3627    8081.72      0      0   4281     7.5    12.1
4        5       Agawam   4015    8181.37      0      0   4826     8.6    21.1
5        7     Amesbury   4273    7037.22      0      0   4824     6.1    16.8
6        8      Amherst   5183   10595.80   6235      0   6454     7.7    17.2
  lunch stratio income score4 score8  salary   english
1  11.8    19.0 16.379    714    691 34.3600 0.0000000
2   2.5    22.6 25.792    731     NA 38.0630 1.2461059
3  14.1    19.3 14.040    704    693 32.4910 0.0000000
4  12.1    17.9 16.111    704    691 33.1060 0.3225806
5  17.4    17.5 15.423    701    699 34.4365 0.0000000
6  26.8    15.7 11.144    714     NA      NA 3.9215686

The dataset contains 16 variables, but we’re only interested in the following:

Variable Description
score8 8th grade scores
stratio Student-teacher ratio
english % English learners
lunch % Receiving lunch subsidy
income Average district income
  1. Fit a model to explain the relationship between percentage of English learners and average 8th grade scores.
  2. Plot the model and the regression line.
  3. Check for correlation between the variables listed above to see if there are other variables that could be included in the model. - HINT: There are some missing values in the dataset. Remove the NA’s from the dataset like we have seen previously.
  4. Add two variables from the list above that are highly correlated with both the independent and the dependent variable and run a linear regression again.
  5. Compare the two models. Did the first model suffer from omitted variable bias?
  6. Plot the residuals against the fitted values from the second model to visually check for heteroskedasticity.
  7. Run the Breusch-Pagan Test to see whether the model suffers from heteroskedastic errors.