In this lab, you'll run simulations to continue to investigate the relationship between effect size, p-values, power, and sample size!
You will be able to:
Remember that the underlying question behind all hypothesis tests is:
"What is the probability I would see this effect due to random fluctuations if there was actually no effect?"
This is exactly what a p-value represents: the chance that the observed data would satisfy the null hypothesis. As such, if the p-value is sufficiently low, you can declare the results statistically significant and reject the null hypothesis. Recall that this threshold is defined as \( \alpha \) , and is also the rate of type I errors. In this lab, you'll investigate the robustness of p-values and their relation with effect-size and sample-size.
To start, import the functions stored in the flatiron_stats.py
file. It contains the stats functions that you previously coded in the last lab: welch_t(a,b)
, welch_df(a, b)
, and p_value(a, b, two_sided=False)
. You'll then use these functions below to further investigate the relationship between p-values and sample size.
# Your code here; import the contents from flatiron_stats.py
# You may also wish to open up flatiron_stats.py in a text editor to preview its contents
from flatiron_stats import *
Before you start running simulations, it will be useful to have a helper function that will generate random samples. Write such a function below which generates 2 random samples from 2 normal distributions. The function should take 6 input parameters:
n1 - The sample size for sample 1
m2 - The underlying population mean for sample 2
import numpy as np
def generate_samples(m1, s1, n1, m2, s2, n2):
# Your code here; have the function create two random samples using the input parameters
sample1 = np.random.normal(loc=m1, scale=s1, size=n1)
sample2 = np.random.normal(loc=m2, scale=s2, size=n2)
return sample1, sample2
For your first simulation, you're going to investigate how the p-value of an experiment relates to sample size when both samples are from identical underlying distributions. To do this, use your generate_samples()
function along with the p_value_welch_ttest()
function defined in the flatiron_stats
file. Use sample sizes from 5 to 750. For each sample size, simulate 100 experiments. For each of these experiments, generate 2 samples of the given sample size. Each sample should have a standard deviation of 1. The first sample should have a mean of 5 and the second should have a mean of 5 plus the effect size, you hope to detect. Calculate the corresponding p-values for a Welch's t-test for each of these sample pairs. Finally, use the p-values to calculate the power of the test. Remember that for all of the simulations where the effect size does not equal zero, the null hypothesis is not true. Store the overall power from the 100 simulations along with the corresponding sample size and effect size. Use varying effect sizes such as [0, 0.01, 0.1, 0.2, 0.5, 1, 2]. You'll then plot power vs sample size for various effect sizes.
import pandas as pd
import matplotlib.pyplot as plt
import seaborn as sns
%matplotlib inline
sns.set_style('darkgrid')
# Your code here
p_vs_e_n = {}
alpha = 0.05
for effect_size in [0, 0.01,.1,.2, .5, 1, 2]:
n_vs_p = {}
for n in range(5,750,5):
p_vals = []
for i in range(100):
sample1, sample2 = generate_samples(m1=5, s1=1, n1=n, m2=5+effect_size, s2=1, n2=n)
p = p_value_welch_ttest(sample1, sample2)
p_vals.append(p)
n_vs_p[n] = np.sum([1 for p in p_vals if p<alpha])/len(p_vals)
p_vs_e_n[effect_size] = n_vs_p
Now that you've simulated the data, go ahead and graph it! Label the x-axis sample size, the y-axis power, and be sure to include a legend for the various effect sizes.
df = pd.DataFrame.from_dict(p_vs_e_n)
df.plot(figsize=(10, 8))
plt.title('Power vs Sample Size for Varying Effect sizes')
plt.xlabel('Sample Size')
plt.ylabel('Power')
plt.legend(title='Effect Size', loc=(1, 0.5))
As you can see, it's also typically incredibly difficult (if not impossible) to accurately detect effect sizes below 0.1!
This lesson summarizes and further builds upon the ideas that we saw in the previous labs. We learned how p-value can be described as a function of effect size and for a given effect size, the p-value may get lower if we increase the sample size considerably. We also saw how p-value alone can not be used in order to identify some results as truly significant, as this can be achieved when there is not a significant effect size.