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
.
Now load the libraries.
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.
========================
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.
Now compare it to the first model.
===================================
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
?
[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):
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.
If you wanted to estimate the residuals from the fitted values manually, you could do so:
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.
[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.
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.
`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.
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. |
======================
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.
- Run a bivariate regression model with
diff
as the dependent variable andtreat
as the predictor. - Control for additional variables. Discuss your choice of variables and refer to omitted variable bias.
- Discuss the improvement in model fit and use the appropriate test statistic.
- Check for non-linearity and if there is solve the problem.
- Check for heteroskedasticity and if there is solve the problem.
- 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 |
- Run a bivariate model to test whether blue-collar voters voted for the Nazis.
- Control for additional variables. Discuss your choice of variables and refer to omitted variable bias.
- Can you fit a model using
shareblue
,sharewhite
,shareself
,sharedomestic
, andshareunemployed
as predictors? Why or why not? - Check for heteroskedasticity and if there is solve the problem.
- 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 |
- Fit a model to explain the relationship between percentage of English learners and average 8th grade scores.
- Plot the model and the regression line.
- 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.
- 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.
- Compare the two models. Did the first model suffer from omitted variable bias?
- Plot the residuals against the fitted values from the second model to visually check for heteroskedasticity.
- Run the Breusch-Pagan Test to see whether the model suffers from heteroskedastic errors.