Bootstrap Monte Carlo Comparison#
In this juypter notebook, we perform a Monte Carlo exercise to illustrate the importance of using the cluster robust variant of the bootstrap when data within clusters is correlated.
The main idea is to repeatedly draw clustered samples, get both uniform and clustered bootstrap estimates in these samples, and then compare how often the true null hypothesis is rejected.
Data Generating Process#
The true data generating process is given by
where the independent variable \(x_{i,g} = x_i + x_g\) and the noise term \(\epsilon_{i,g} = \epsilon_i + \epsilon_g\) each consist of an individual and a cluster term.
In the simulations we perform below, we have \(\beta_0 = \beta_1 =0\). \(x_i\) and \(x_g\) are drawn from a standard normal distribution, and \(\epsilon_i\) and \(\epsilon_g\) are drawn from a normal distribution with \(\mu_0\) and \(\sigma=0.5\). The value of \(\sigma\) is chosen to not blow up rejection rates in the independent case too much.
import estimagic as em
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
import scipy
import statsmodels.api as sm
from joblib import Parallel, delayed
def create_clustered_data(nclusters, nobs_per_cluster, true_beta=0):
"""Create a bivariate clustered dataset with specified number of
clusters and number of observations per cluster that has a population
value of true_beta for the logit coefficient on the independent variable.
Args:
nclusters (int): Number of clusters.
nobs_per_cluster (int): Number of observations per cluster.
true_beta (int): The true logit coefficient on x.
Returns:
pd.DataFrame: Clustered dataset.
"""
x_cluster = np.random.normal(size=nclusters)
x_ind = np.random.normal(size=nobs_per_cluster * nclusters)
eps_cluster = np.random.normal(size=nclusters, scale=0.5)
eps_ind = np.random.normal(size=nobs_per_cluster * nclusters, scale=0.5)
y = []
x = []
cluster = []
for g in range(nclusters):
for i in range(nobs_per_cluster):
key = (i + 1) * (g + 1) - 1
arg = (
true_beta * (x_cluster[g] + x_ind[key]) + eps_ind[key] + eps_cluster[g]
)
y_prob = 1 / (1 + np.exp(-arg))
y.append(np.random.binomial(n=1, p=y_prob))
x.append(x_cluster[g] + x_ind[(i + 1) * (g + 1) - 1])
cluster.append(g)
y = np.array(y)
x = np.array(x)
cluster = np.array(cluster)
return pd.DataFrame({"y": y, "x": x, "cluster": cluster})
Monte Carlo Simulation Code#
The following function computes bootstrap t-values. As suggested my Cameron and Miller (2015), critical values are the 0.975 quantiles from a t distribution with n_clusters
-1 degrees of freedom.
def get_t_values(data, sample_size=200, hyp_beta=0, cluster=False):
"""Get bootstrap t-values for testing the hypothesis that beta == hyp_beta.
Args:
data (pd.DataFrame): Original dataset.
sample_size (int): Number of bootstrap samples to draw.
hyp_beta (float): Hypothesised value of beta.
cluster (bool): Whether or not to cluster on the cluster column.
Returns:
float: T-Value of hypothesis.
"""
def logit_wrap(df):
y = df["y"]
x = df["x"]
result = sm.Logit(y, sm.add_constant(x)).fit(disp=0).params
return pd.Series(result, index=["constant", "x"])
if cluster is False:
result = em.bootstrap(data=data, outcome=logit_wrap, n_draws=sample_size)
estimates = pd.DataFrame(result.outcomes)["x"]
else:
result = em.bootstrap(
data=data,
outcome=logit_wrap,
n_draws=sample_size,
cluster_by="cluster",
)
estimates = pd.DataFrame(result.outcomes)["x"]
return (estimates.mean() - hyp_beta) / estimates.std()
def monte_carlo(nsim, nclusters, nobs_per_cluster, true_beta=0, n_cores=8):
"""Run a simulation for rejection rates and a logit data generating process.
Rejection rates are based on a t distribution with nclusters-1 degrees of freedom.
Args:
nsim (int): Number of Monte Carlo draws.
nclusters (int): Number of clusters in each generated dataset.
nobs_per_cluster (int) Number of observations per cluster.
true_beta (int): Population value of logit coefficient on x.
n_cores (int): Number of jobs for Parallelization.
Returns:
pd.DataFrame: DataFrame of average rejection rates.
"""
np.zeros(nsim)
np.zeros(nsim)
def loop():
df = create_clustered_data(nclusters, nobs_per_cluster, true_beta)
return [get_t_values(df), get_t_values(df, cluster=True)]
t_value_array = np.array(
Parallel(n_jobs=n_cores)(delayed(loop)() for _ in range(nsim))
)
t_value_array = np.array([loop() for _ in range(nsim)])
crit = scipy.stats.t.ppf(0.975, nclusters - 1)
result = pd.DataFrame(np.abs(t_value_array) > crit, columns=["uniform", "cluster"])
return result
Results#
Here, we perform Monte Carlo simulations with the above functions. In each simulation, the sample size is 200, but the number of clusters varies across simulations. Be warned that the code below takes a long time to run.
np.random.seed(505)
results_list = []
for g, k in [[20, 50], [100, 10], [500, 2]]:
results_list.append(monte_carlo(nsim=100, nclusters=g, nobs_per_cluster=k))
mean_rejection_data = pd.DataFrame([x.mean() for x in results_list])
mean_rejection_data["nclusters"] = [20, 100, 500]
mean_rejection_data.set_index("nclusters", inplace=True)
y = mean_rejection_data
x = y.index.values
plt.rcParams["figure.figsize"] = (8, 5)
plt.xlabel("Number of clusters", fontsize=12)
plt.ylabel("Rejection rate", fontsize=12)
plt.plot(x, y["uniform"], label="Uniform Bootstrap", color="blue", marker="o")
plt.plot(x, y["cluster"], label="Cluster Bootstrap", color="red", marker="o")
plt.legend()
plt.suptitle("Comparison of Rejection Rates", fontsize=15)
Text(0.5, 0.98, 'Comparison of Rejection Rates')
We can see that when the number of clusters is low, it is particularly important to use the cluster robust bootstrap, since rejection with the regular bootstrap is excessive. For a large number of clusters, clustering naturally becomes less important.