Regression Discontinuity Designs
Welcome
Introduction!
Welcome to our seventh tutorial for the Statistics II: Statistical Modeling & Causal Inference (with R) course.
During this week’s lecture you were introduced to Regression Discontinuity Designs (RDDs).
In this lab session we will:
- Leverage visualizations with
ggplot2
to explore our discontinuity setups - Learn how to model our discontinuity setups under different functional forms with
lm()
- Learn how to model our discontinuity setups under different functional forms with
rdrobust::rdrobust()
Packages
# These are the libraries we will use today. Make sure to install them in your console in case you have not done so previously.
library(dplyr) # for data wrangling
library(ggplot2) # for creating plots
library(rdrobust) # for rdrobust()
library(readr) # for loading the .csv data
set.seed(42) # for consistent results
mlda_df <- readr::read_csv("https://raw.githubusercontent.com/seramirezruiz/stats-ii-lab/master/Session%206/data/mlda.csv") # loading data from Mastering Metrics
Example 1. Measuring the effect of the minimum legal drinking age (MLDA) on mortality
In an effort to address the social and public health problems associated with underage drinking, a group of American college presidents have lobbied states to return the minimum legal drinking age (MLDA) to the Vietnam era threshold of 18. The theory behind this effort (known as the Amethyst Initiative) is that legal drinking at age 18 discourages binge drinking and promotes a culture of mature alcohol consumption. This contrasts with the traditional view that the age-21 MLDA, while a blunt and imperfect tool, reduces youth access to alcohol, thereby preventing some harm. Angrist and Pischke (2014)
You have been hired as an outside consultant by Mothers Against Drunk Driving (MADD) to study whether the over-21 drinking minimum in the US helps reduce alcohol consumption by young adults and its harms, or is it just not working. This example is based on data from Carpenter and Dobkin (2011).
Checking visually whether a sharp-RDD makes sense for the analysis
We want to know whether our threshold is in fact the cut-off for treatment. In this case, the law is pretty clear: young adults in the US can legally drink at age 21.
ggplot(mlda_df, aes(x = agecell, # actual age
y = treatment, # are they over 21 or not
color = factor(treatment))) +
geom_point() +
labs(x = "Age",
y = "Treatment Probability") +
scale_color_discrete(name = " ",
labels = c("Under legal drinking age", "Over legal drinking age")) +
geom_vline(xintercept = 21, linetype = "dotted") + # NEW GEOM A VERTICAL LINE!
theme_minimal()
We can see from the graph that at the 21-years-of-age threshold, young adults can legally consume and buy alcohol in the US, which would make age a viable forcing variable for a sharp-RDD set-up.
Running our regression models
As was pointed out in the lecture, we must decide on a model that we believe reflects the relationship of our \(E(Y_i|\tilde{X}_i)\):
- linear, common slope
- linear, different slopes
- non-linear
Remember that each model corresponds to a particular set of assumptions
We will show you how to visualize this with ggplot
.
LET’S LOOK AT A SCATTERPLOT TO GET AN IDEA OF WHAT WE ARE DEALING WITH
ggplot(mlda_df,
aes(x = agecell, # age
y = outcome)) + # mortality rate per 100k accidents
geom_point() +
geom_vline(xintercept = 21, linetype = "dotted") +
labs(title = "Exploratory plot",
x = "Forcing variable (Age)",
y = "Mortality rate from motor vehicle \naccidents (per 100,000)") +
theme_minimal()
NOTE that we have the variable forcing
in this dataset, which is centered at the cutoff. It is nothing but the variable age-21
Linear model with common slopes
Let’s run a linear model with common slopes and plot our results.
NOTE that the forcing variable in this case (age) is CENTERED at 0 (age 21) and is the distance from age 21 in years, while treatment is just binary over/under 21.
# running linear model with common slope
linear_common_slope <- lm(outcome ~ treatment + forcing, data = mlda_df)
stargazer::stargazer(linear_common_slope, type = "text")
##
## ===============================================
## Dependent variable:
## ---------------------------
## outcome
## -----------------------------------------------
## treatment 4.534***
## (0.768)
##
## forcing -3.149***
## (0.337)
##
## Constant 29.356***
## (0.429)
##
## -----------------------------------------------
## Observations 48
## R2 0.703
## Adjusted R2 0.689
## Residual Std. Error 1.329 (df = 45)
## F Statistic 53.142*** (df = 2; 45)
## ===============================================
## Note: *p<0.1; **p<0.05; ***p<0.01
WHAT DO THESE RESULTS TELL US?
In line with our assumptions for linear models with common slope, we consider that treatment effect \(D_i\) does not depend on the forcing \(X_i\). We can formalize this model as: \[E(Y_i|X_i,D_i) = \tilde{\beta_0} + \beta_1 D_i + \beta_2\tilde{X}_i\] Hence we can say, that given our \(\beta_1\) we can expect 4.53 more deaths from motor vehicle accidents per 100,000 for those who can legally drink. We also see that for every year of age increase, the number of expected deaths per 100,000 decreases by 3.15. (Our \(\beta_2 = -3.1488\)).
We can graph our results with ggplot()
by extracting the predicted values of the model to recreate the linear fit:
mlda_df$yhat_linear <- predict(linear_common_slope) # we create a new variable containing the predicted mortality rate
linear_plot <- mlda_df %>% # for this plot make sure to put the df outside the ggplot() and pipe it
ggplot(aes(x = forcing,
y = yhat_linear, # notice here the predicted y
col = factor(treatment))) +
geom_point(aes(x = forcing,
y = outcome, # notice here the actual outcome
col = factor(treatment))) +
geom_vline(xintercept = 0, linetype = "dotted") +
labs(title = "Linear model with common slope",
x = "Forcing variable (Age)",
y = "Mortality rate from motor vehicle \naccidents (per 100,000)") +
geom_line(data = mlda_df[mlda_df$forcing >= 0,],
color = "#cc0055", # color lines
size = 1) +
geom_line(data = mlda_df[mlda_df$forcing < 0,],
color = "#696969", # color lines
size = 1) +
scale_color_manual(name = "",
values = c("#696969", "#cc0055"),
labels = c("Control", "Treatment")) + #change colors manually of color argument in aes()
theme_minimal()
linear_plot
Linear model with different slopes
Let’s run the linear model to gather the slopes for both groups and plot our results. This is achieved by interacting our treatment and forcing variables.
linear_different_slope <- lm(outcome ~ treatment*forcing, data = mlda_df)
stargazer::stargazer(linear_different_slope, type = "text")
##
## ===============================================
## Dependent variable:
## ---------------------------
## outcome
## -----------------------------------------------
## treatment 4.534***
## (0.751)
##
## forcing -2.568***
## (0.466)
##
## treatment:forcing -1.162*
## (0.659)
##
## Constant 29.929***
## (0.531)
##
## -----------------------------------------------
## Observations 48
## R2 0.722
## Adjusted R2 0.703
## Residual Std. Error 1.299 (df = 44)
## F Statistic 38.125*** (df = 3; 44)
## ===============================================
## Note: *p<0.1; **p<0.05; ***p<0.01
WHAT DO THESE RESULTS TELL US?
In line with our assumptions for linear models with different slope, we allow our treatment effect \(D_i\) to vary along the forcing \(X_i\). We can formalize this model as:
\[E(Y_i|X_i,D_i) = \tilde{\beta_0} + \beta_1D_i+ \beta_2X_i + \tilde{\beta_3}D_i\tilde{X}_i\]
Hence we can say, that at given our \(\beta_1\), we can expect 4.53 more deaths from motor vehicle accidents per 100,000 for those who can legally drink at the threshold. Now we have two different slopes for year-of-age changes. For under-21 individuals, an increase of one year of age would on average result in 2.57 less deaths from motor vehicle accidents (our \(\beta_2 = -2.5676\)). For those of legal drinking age, we would expect 3.73 less deaths per 100,000 for every one year of age increase (our \(\beta_2X_i + \tilde{\beta_3}D_i\tilde{X}_i = -2.5676 + (-1.1624) = - 3.73\)).
We can graph our results with ggplot
by just adding a smooth geom. Since we have added treatment to our color aesthetic, ggplot()
will automatically create the regression line for each group
diff_slopes_plot <- mlda_df %>%
ggplot(aes(x = forcing,
y = outcome,
col = factor(treatment))) +
geom_point() +
geom_vline(xintercept = 0, linetype = "dotted") +
geom_smooth(method = "lm", se = F) + # NORMAL SMOOTH
labs(title = "Linear model with different slopes",
x = "Forcing variable (Age)",
y = "Mortality rate from motor vehicle \naccidents (per 100,000)") +
scale_color_manual(name = "",
values = c("#696969", "#cc0055"),
labels = c("Control", "Treatment")) + #change colors manually of color argument in aes()
theme_minimal()
diff_slopes_plot
Question
Where can you see our \(\beta_0\), \(\beta_1\), and \(\beta_2\) in the previous plot?
Non-linear model
Let’s run a quadratic model and plot our results.
THIS IS HOW WE WOULD FORMALIZE A QUADRATIC MODEL
\[E(Y_i∣X_i, D_i) = \gamma_0 + + \tau_1D_i + \gamma_2\tilde{X_i} + \gamma_3\tilde{X^2_i} + \alpha_1\tilde{X_i}D_i + \alpha_2\tilde{X^2_i}D_i\]
We can input this in our regression model as follows:
quadratic <- lm(outcome ~ forcing +
I(forcing^2) + # I tells R to interpret "as is"
treatment +
I(forcing * treatment) +
I((forcing^2) * treatment),
data = mlda_df)
stargazer::stargazer(quadratic, type = "text")
##
## =====================================================
## Dependent variable:
## ---------------------------
## outcome
## -----------------------------------------------------
## forcing -2.933
## (1.914)
##
## I(forcing2) -0.185
## (0.940)
##
## treatment 4.663***
## (1.155)
##
## I(forcing * treatment) -0.823
## (2.706)
##
## I((forcing2) * treatment) 0.198
## (1.329)
##
## Constant 29.809***
## (0.817)
##
## -----------------------------------------------------
## Observations 48
## R2 0.722
## Adjusted R2 0.689
## Residual Std. Error 1.329 (df = 42)
## F Statistic 21.864*** (df = 5; 42)
## =====================================================
## Note: *p<0.1; **p<0.05; ***p<0.01
WHAT DO THESE RESULTS TELL US?
In line with our assumptions for non-linear models, we allow our treatment effect \(D_i\) to vary along the forcing \(X_i\). In this case with quadratic interactions. We can formalize this model as:
\[E(Y_i∣X_i, D_i) = \gamma_0 + + \tau_1D_i + \gamma_2\tilde{X_i} + \gamma_3\tilde{X^2_i} + \alpha_1\tilde{X_i}D_i + \alpha_2\tilde{X^2_i}D_i\]
Hence we can say, that at given our \(\tau\), we can expect 4.66 more deaths from motor vehicle accidents per 100,000 for those who can legally drink at the threshold. We could also calculate the expected value of \(Y\) at different levels of \(X\).
Say we want to know what the expected value is for 23-year-olds. Since our forcing variable is 0 at 21 years of age, we can think of 23 as 2. Additionally, 23-year-olds are above the legal drinking age minimum, therefore for them the value of \(D\) is 1.
\[E(Y | X=2, D =1) = 29.8090 + 4.6629(1) - 2.9330(2) -0.1852(2)^2 - 0.8231 (2*1) + 0.1985(2*1)^2 = 27.01\] Based on this, we would expect a mortality rate from motor vehicle accidents of 27.01 per 100,000 for 23-year-olds.
We can graph our results with ggplot
by extracting the predicted values of our quadratic model to recreate the fit:
mlda_df$yhat_quadratic <- predict(quadratic)
quadratic_plot <- mlda_df %>%
ggplot(aes(x = forcing,
y = yhat_quadratic, #note predicted y
col = factor(treatment))) +
geom_point(aes(x = forcing,
y = outcome,
col = factor(treatment))) +
geom_vline(xintercept = 0, linetype = "dotted") +
labs(title = "Quadratic plot",
x = "Forcing variable (Age)",
y = "Mortality rate from motor vehicle \naccidents (per 100,000)") +
geom_line(data = mlda_df[mlda_df$forcing >= 0,],
color = "#cc0055", # color lines
size = 1) +
geom_line(data = mlda_df[mlda_df$forcing < 0,],
color = "#696969", # color lines
size = 1) +
scale_color_manual(name = "",
values = c("#696969", "#cc0055"),
labels = c("Control", "Treatment")) + #change colors manually of color argument in aes()
theme_minimal()
quadratic_plot
Calculating the LATE with rdrobust()
rdrobust()
is part of the rdrobust
package. It performs local linear regressions to either side of the cutpoint using optimal bandwidth calculation. The syntax is the following:
model <- rdrobust::rdrobust(x,
y,
c = cutoffvalue,
kernel = "tri", #default
bwselect = "mserd") #default
We have the option to set the cutpoint, kernel type, order of the local polynomial, etc.: https://cran.r-project.org/web/packages/rdrobust/rdrobust.pdf
llr <- rdrobust::rdrobust(mlda_df$outcome,
mlda_df$forcing,
c = 0,
kernel = "tri",
bwselect = "mserd")
summary(llr)
## Call: rdrobust
##
## Number of Obs. 48
## BW type mserd
## Kernel Triangular
## VCE method NN
##
## Number of Obs. 24 24
## Eff. Number of Obs. 6 6
## Order est. (p) 1 1
## Order bias (q) 2 2
## BW est. (h) 0.487 0.487
## BW bias (b) 0.738 0.738
## rho (h/b) 0.660 0.660
## Unique Obs. 24 24
##
## =============================================================================
## Method Coef. Std. Err. z P>|z| [ 95% C.I. ]
## =============================================================================
## Conventional 4.901 2.059 2.380 0.017 [0.864 , 8.937]
## Robust - - 1.881 0.060 [-0.198 , 9.674]
## =============================================================================
WHAT DO THESE RESULTS TELL US?
The model is telling us that based on the calculation, the estimated effect would be 4.90 more deaths per 100,000 for those over-21.
The most straight-forward way to graph the output of this model is through the rdrobust::rdplot()
function:
rdrobust::rdplot(mlda_df$outcome,
mlda_df$forcing,
c = 0,
kernel = "tri",
title = "Motor Vehicle Accidents Death",
x.label = "Age from 21",
y.label = "Mortality rate from motor vehicle \naccidents (per 100,000)"
)
Quadratic model with rdrobust()
quadratic_rdrobust <- rdrobust::rdrobust(mlda_df$outcome,
mlda_df$forcing,
c = 0,
kernel = "tri",
bwselect = "mserd",
p = 2) #polynomial 2
summary(quadratic_rdrobust)
## Call: rdrobust
##
## Number of Obs. 48
## BW type mserd
## Kernel Triangular
## VCE method NN
##
## Number of Obs. 24 24
## Eff. Number of Obs. 10 10
## Order est. (p) 2 2
## Order bias (q) 3 3
## BW est. (h) 0.821 0.821
## BW bias (b) 1.074 1.074
## rho (h/b) 0.764 0.764
## Unique Obs. 24 24
##
## =============================================================================
## Method Coef. Std. Err. z P>|z| [ 95% C.I. ]
## =============================================================================
## Conventional 4.778 2.337 2.044 0.041 [0.197 , 9.360]
## Robust - - 1.627 0.104 [-0.911 , 9.811]
## =============================================================================
rdrobust::rdplot(mlda_df$outcome,
mlda_df$forcing,
c = 0,
kernel = "tri",
p = 2,
title = "Motor Vehicle Accidents Death",
x.label = "Age from 21",
y.label = "Mortality rate from motor vehicle \naccidents (per 100,000)"
)
Example 2. Measuring the long term effects of a conditional cash transfer program on educational achievement
Imagine that you work as a technical adviser for the Ministry of Education in your country. You are tasked to assess whether a Conditional Cash Transfer (CCT) program established decades before yields positive results on the beneficiaries’ educational attainment. There is a large amount of evidence which suggests that CCTs encourage households to increase the use of educational services.
You read the guidelines for the program. Families receive a stipend per child provided they keep their them in school and take them for health checks. Additionally, you note that under the rules of the program, beneficiaries are selected based on a household income threshold of €20000. You decide to dive into the data with the idea that a discontinuity is created based on the income threshold. (This example utilizes simulated data)
cct_df <- readr::read_csv("https://raw.githubusercontent.com/seramirezruiz/stats-ii-lab/master/Session%206/data/cct_data.csv") # loading simulated data frame of the program
The dataset consists of:
hh_income
: household income in eurosyears_of_schooling
: years of schooling for respondenttreatment
: binary variable indicating whether respondent was a beneficiary of the program
Checking visually whether a sharp-RDD makes sense for the analysis
What we are looking for in this case is whether our €20000 threshold is in fact the cut-off for treatment. That is to say, that only those who had a household income of equal or less than €20000 received the cash transfer.
ggplot(cct_df,
aes(x = hh_income,
y = years_of_schooling,
color = factor(treatment))) +
geom_point() +
labs(x = "Household Income",
y = "Years of Schooling") +
scale_color_discrete(name = " ",
labels = c("No treatment", "Treatment")) +
geom_vline(xintercept = 20000, linetype = "dotted") +
theme_minimal()
ggplot(cct_df,
aes(x = hh_income,
y = treatment,
color = factor(treatment))) +
geom_point() +
labs(x = "Household Income",
y = "Treatment") +
scale_color_discrete(name = " ",
labels = c("No treatment", "Treatment")) +
geom_vline(xintercept = 20000, linetype = "dotted") +
theme_minimal()
We can see from the graph that our €20000 threshold is in fact cutting off the distribution of the treatment. This would make household income a viable forcing variable for a sharp-RDD set-up.
Estimating our model
We can see that the relationship is fairly linear, so we decide to run a linear model with common slope.
# running linear model with common slope
ed_achievement <- lm(years_of_schooling ~ treatment + hh_income, data = cct_df)
stargazer::stargazer(ed_achievement, type = "text")
##
## ================================================
## Dependent variable:
## ----------------------------
## years_of_schooling
## ------------------------------------------------
## treatment 2.460***
## (0.038)
##
## hh_income 0.001***
## (0.00000)
##
## Constant -2.648***
## (0.111)
##
## ------------------------------------------------
## Observations 5,000
## R2 0.815
## Adjusted R2 0.815
## Residual Std. Error 0.817 (df = 4997)
## F Statistic 11,008.950*** (df = 2; 4997)
## ================================================
## Note: *p<0.1; **p<0.05; ***p<0.01
WHAT DO THESE RESULTS TELL US?
In line with our assumptions for linear models with common slope, we consider that treatment effect, \(D_i\), does not depend on the forcing \(X_i\). Hence, we can expect that students who received the treatment get on average 2.4 more years of schooling. We also see that for every €1,000 increase in the household income, students are expected to attain 0.6274 more years of education. (Our \(\beta = -6.274e-04*1000\)).
Getting familiar with LOESS
Locally weighted smoothing is a popular tool used in regression analysis that creates a smooth line through a scatter plot to help you to see relationship between variables and foresee trends. We can introduce it to our ggplot()
as a part of geom_smooth by calling method “loess”.
ggplot(cct_df,
aes(x = hh_income,
y = years_of_schooling,
color = factor(treatment))) +
geom_point(alpha = 0.1) +
labs(x = "Household Income",
y = "Years of Schooling") +
geom_smooth(method = "loess") + # instead of lm, we use loess. See the difference? try with lm
scale_color_discrete(name = " ",
labels = c("No treatment", "Treatment")) +
geom_vline(xintercept = 20000, linetype = "dotted") +
theme_minimal()
The LOESS smoothing is not very visible in this relationship because of the way we defined the simulated data. Let’s look at how it would look in our drinking age example:
ggplot(mlda_df,
aes(x = agecell,
y = outcome,
col = factor(treatment))) +
geom_point() +
geom_smooth(method = "loess") +
labs(title = "LOESS smoothing",
x = "Forcing variable (Age)",
y = "Mortality rate from motor vehicle \naccidents (per 100,000)") +
scale_color_manual(name = "",
values = c("#F8766D", "#00BFC4"),
labels = c("Control", "Treatment")) +
theme_minimal()
Violations to the assumptions
You are made aware by a tax expert from your unit that €20000 is the upper-boundary for a very well known tax concession. You are afraid that people may be sorting themselves before the household income cut-off to become beneficiaries of multiple programs. You decide to check your data.
ggplot(cct_df,
aes(x = hh_income)) +
geom_histogram(bins = 50, fill = "#cc0055") +
labs(title = "Income distribution",
x = "Household Income",
y = "Number of respondents") +
geom_vline(xintercept = 20000, linetype = "dotted") +
theme_minimal()
This case looks a bit ambiguous. Do you think people are sorting out just before the cut-off? If sorting were to exist which assumptions would be challenged? Would the existence of other programs that have the same threshold affect a causal reading of our results?
There are a couple of tests researchers can employ. We will learn two ways. First, a method by which the research chooses a window of sorting to check if the distribution could have occurred by chance and second the McCrary test you met in Cunningham (2021).
Binomial test
When we apply an exact binomial test. Our interest is to see whether the distribution around the threshold could exist by chance. In this case, let’s check ±500 and ±250 around the threshold.
To gather only the units that reported household incomes from €19500 to €20500, we will use a new function dplyr::between()
.
dplyr::between()
is a shortcut for x >= left & x <= right
. Let’s look at it at work.
cct_df %>%
dplyr::filter(dplyr::between(hh_income, 19500, 20500)) %>% #filter values between 19500 and 20500
dplyr::group_by(treatment) %>%
dplyr::summarize(n = n()) %>%
knitr::kable() %>%
kableExtra::kable_styling()
treatment | n |
---|---|
0 | 255 |
1 | 300 |
We have 300 units just below the threshold and 255 units just above. We can use this information to run our exact binomial test. We seek to understand if the observed distributions deviate from expected distribution of observations into the two categories.
We can do the same for €19750 to €20250
cct_df %>%
dplyr::filter(dplyr::between(hh_income, 19750, 20250)) %>% #filter values between 19750 and 20250
dplyr::group_by(treatment) %>%
dplyr::summarize(n = n()) %>%
knitr::kable() %>%
kableExtra::kable_styling()
treatment | n |
---|---|
0 | 115 |
1 | 175 |
binom.test(one_of_the_n, total_n, p = probability_of_success)
Let’s see what this would say for ±500:
binom.test(300, 555, p = 0.5)
##
## Exact binomial test
##
## data: 300 and 555
## number of successes = 300, number of trials = 555, p-value = 0.06171
## alternative hypothesis: true probability of success is not equal to 0.5
## 95 percent confidence interval:
## 0.4980565 0.5825909
## sample estimates:
## probability of success
## 0.5405405
WHAT DO THESE RESULTS TELL US?
According to the test, we see that the observed distributions do not deviate from expected distribution of observations into the two categories when we expect that units just around the threshold end up on either group by chance (coin flip logic, i.e., p = 0.5). In other words, this results do not present some evidence of sorting in this window.
How about ±250?:
binom.test(115, 290, p = 0.5)
##
## Exact binomial test
##
## data: 115 and 290
## number of successes = 115, number of trials = 290, p-value = 0.0005095
## alternative hypothesis: true probability of success is not equal to 0.5
## 95 percent confidence interval:
## 0.3398404 0.4553997
## sample estimates:
## probability of success
## 0.3965517
WHAT DO THESE RESULTS TELL US?
According to the test, we see that the observed distributions deviate from expected distribution of observations into the two categories when we expect that units just around the threshold end up on either group by chance (coin flip logic, i.e., p = 0.5). In other words, this results present some evidence of sorting in this window.
McCrary Sorting test
An alternative way to check for self-sorting is the McCrary Sorting test. In this test, the discretion on window selection is taken away from the researcher (at least in the defaults). The McCrary Sorting test is included in the rdd
package. This is the syntax of the test:
rdd::Dcdensity(runvar=running_variable, cutpoint=cutpoint)
Let’s see it in practice:
rdd::DCdensity(cct_df$hh_income, cutpoint = 20000)
## [1] 0.04168422
WHAT DO THESE RESULTS TELL US?
The default output is a p-value of the test. A p-value below the significance threshold indicates that the user can reject the null hypothesis of no sorting. In other words, this test would suggest that our observed distributions deviate from the expected distribution of observations. This results present some evidence of sorting.