X-Git-Url: https://code.communitydata.science/stats_class_2019.git/blobdiff_plain/debb8396f68285dde7bfcff6cc278498cf648a6e..f2036e416b850a6c3cd936db4574de3dd466ee27:/problem_sets/week_06/ps6-worked-solution.html diff --git a/problem_sets/week_06/ps6-worked-solution.html b/problem_sets/week_06/ps6-worked-solution.html index 7a80c67..d182029 100644 --- a/problem_sets/week_06/ps6-worked-solution.html +++ b/problem_sets/week_06/ps6-worked-solution.html @@ -13,202 +13,17 @@ -
PC0. First we import the data.
-raw_df = read.csv("~/Desktop/DeleteMe/Teaching/owan03.csv") # Note that I saved the file as a CSV for importing to R
+Note that Jeremy created this solution set with some code that uses functions drawn from the tidyverse
. Wherever possible, we have also provided solution code using base R functions or alternatives since the course has not really provided an introduction to the tidyverse.
+
+PC0
+First we import the data.
+raw_df = read.csv("/tmp/owan03.csv") # Note that I saved the file as a CSV for importing to R
head(raw_df)
## X1 X2 X3 X4
## 1 70 49 30 34
@@ -326,35 +215,43 @@ head(raw_df)
## 4 87 67 65 48
## 5 92 70 76 65
## 6 93 74 83 91
-PC1. Letâs reshape the data
-library(tidyverse)
-## ââ Attaching packages ââââââââââââââââââââââââââââââââââ tidyverse 1.2.1 ââ
-## â ggplot2 3.1.0 â purrr 0.2.5
-## â tibble 2.1.1 â dplyr 0.7.7
-## â tidyr 0.8.2 â stringr 1.3.1
-## â readr 1.1.1 â forcats 0.3.0
-## ââ Conflicts âââââââââââââââââââââââââââââââââââââ tidyverse_conflicts() ââ
-## â dplyr::filter() masks stats::filter()
-## â dplyr::lag() masks stats::lag()
+
+
+PC1
+Letâs organize the data. We want this in âlongâ format where the survival times in weeks for each of the three dosage levels are all arranged in a single column and another variable labels the dosage level. There are (surprise!) many ways to do this. Here are two examples using a function called melt()
from a library called reshape
and another function called gather()
from the tidyverse
library. Note that youâll need to install the respective library and load it before you can call either function!
+library(reshape)
+library(tidyr)
+##
+## Attaching package: 'tidyr'
+## The following objects are masked from 'package:reshape':
+##
+## expand, smiths
# Rename the columns
colnames(raw_df) <- c("None", "Low", "Medium","High")
-# Gather gets all of the columns from the dataframe and
-# use them as keys, with the value as the value from that cell.
-# This gives us a "long" dataframe
-df <- gather(raw_df, dose, weeks_alive)
+# Here's a version using "melt"
+d <- melt(raw_df, na.rm=TRUE)
+## Using as id variables
+names(d) <- c("dose", "weeks_alive")
-# We want to treat dose as a factor, so we can group by it easier
-df$dose <- as.factor(df$dose)
+# "gather" similarly takes all of the columns from the dataframe and uses them as keys, with the value as the value from that # cell. This gives us a "long" dataframe
+df <- gather(raw_df, dose, weeks_alive)
+# We want to treat dose as a factor, so we can group by it easier. Specifying the levels helps with labels later on.
+df$dose <- factor(df$dose, levels=c("None", "Low", "Medium", "High"))
+# Finally, remove NAs
+df <- df[complete.cases(df),]
-# Finally, when we look at it, we see some missing data (NA); this is because not all group sizes were the same.
-# We can safely remove these.
-df <- df[complete.cases(df),]
-PC2: Now weâre goint to get statistics and create some visualizations
-tapply(df$weeks_alive, df$dose, summary)
-## $High
+# Uncomment this to see that the results are the same:
+## df==d
+
+
+PC2
+Now weâre going to calculate summary statistics and create some visualizations
+# First, using the tapply function
+tapply(df$weeks_alive, df$dose, summary)
+## $None
## Min. 1st Qu. Median Mean 3rd Qu. Max.
-## 34.00 45.00 56.50 65.25 92.75 102.00
+## 70.00 85.00 93.00 91.36 101.00 103.00
##
## $Low
## Min. 1st Qu. Median Mean 3rd Qu. Max.
@@ -364,73 +261,100 @@ df <- df[complete.cases(df),]
## Min. 1st Qu. Median Mean 3rd Qu. Max.
## 30.00 58.25 79.50 71.50 89.25 97.00
##
-## $None
+## $High
## Min. 1st Qu. Median Mean 3rd Qu. Max.
-## 70.00 85.00 93.00 91.36 101.00 103.00
-# Alternative way to do this using tidyverse
-
-df %>% group_by(dose) %>% summarize_all(c('min','max','mean', 'IQR'))
+## 34.00 45.00 56.50 65.25 92.75 102.00
+# Alternative way to do this using tidyverse "group_by" function:
+library(dplyr)
+##
+## Attaching package: 'dplyr'
+## The following object is masked from 'package:reshape':
+##
+## rename
+## The following objects are masked from 'package:stats':
+##
+## filter, lag
+## The following objects are masked from 'package:base':
+##
+## intersect, setdiff, setequal, union
+df %>% group_by(dose) %>% summarize_all(c('min','max','mean', 'IQR'))
## # A tibble: 4 x 5
## dose min max mean IQR
## <fct> <dbl> <dbl> <dbl> <dbl>
-## 1 High 34 102 65.2 47.8
+## 1 None 70 103 91.4 16
## 2 Low 49 89 69.9 14
## 3 Medium 30 97 71.5 31
-## 4 None 70 103 91.4 16
+## 4 High 34 102 65.2 47.8
When it comes to visualizations, we definitely want to use ggplot. We have lots of options for what to do.
+library(ggplot2)
+## Registered S3 methods overwritten by 'ggplot2':
+## method from
+## [.quosures rlang
+## c.quosures rlang
+## print.quosures rlang
# Histograms
-h_plot = df %>% ggplot(aes(x=weeks_alive, # What to summarize
- fill = dose # How to group by color
- )) + geom_histogram(position = 'dodge', bins = 5)
+h_plot <- ggplot(data=df, aes(x=weeks_alive, fill = dose)) + geom_histogram(position = 'dodge', bins = 5)
h_plot
-
+
# In this case, faceted histograms is probably better
-
-h_facet = df %>% ggplot(aes(x=weeks_alive, # What to summarize
- )) + geom_histogram(bins = 5) + facet_grid(~dose)
-
+h_facet <- ggplot(data=df, aes(x=weeks_alive)) + geom_histogram(bins = 5) + facet_grid(~dose)
h_facet
-
-# Density plots
-d_plot = df %>% ggplot(aes(x=weeks_alive, # What to summarize
- fill = dose # How to group by color
- )) + geom_density(alpha = .2)
+
+# Even easier to compare if we flip the coordinates (rotate the axes)
+h_facet+coord_flip()
+
+# Density plots. Note that setting a value <1 for "alpha" makes the fill more transparent
+d_plot <- ggplot(data=df, aes(x=weeks_alive, fill = dose)) + geom_density(alpha = .2)
d_plot
-
+
# Boxplots
-box_plot = df %>% ggplot(aes(y=weeks_alive,
- x = dose
- )) + geom_boxplot()
+box_plot <- ggplot(data=df, aes(y=weeks_alive, x = dose)) + geom_boxplot()
box_plot
-
-# My favorite - ridgeline plots
+
+# Jeremy's favorite - ridgeline plots
# install.packages('ggridges')
-
library(ggridges)
##
## Attaching package: 'ggridges'
## The following object is masked from 'package:ggplot2':
##
## scale_discrete_manual
-ridge_plot = df %>% ggplot(aes(x=weeks_alive, y = dose)) +
- geom_density_ridges(jittered_points = T)
+ridge_plot <- ggplot(data=df, aes(x=weeks_alive, y = dose)) + geom_density_ridges(jittered_points = T, fill = 'orange')
ridge_plot
## Picking joint bandwidth of 10.5
-
-Itâs a bit tough to tell, but the overall assumptions of normality and equal variance seem reasonable.
+
+# add a fancy minimalist theme to make it prettier:
+ridge_plot + theme_minimal()
+## Picking joint bandwidth of 10.5
+
+A two sample t-test assumes independence and normality. An ANOVA assumes independence, normality, and equal variance. Itâs a bit tough to tell, but the overall assumption of equal variance seems reasonable. Normality is a bit of a hard sell within groups or overall. Nevertheless, most analysts would march ahead with the analysis despite these violations of assumptions. We can discuss how you might think and talk about this in class.
The global mean is
mean(df$weeks_alive)
## [1] 75.55263
-PC3. T-test between None and Any, and between None and High.
-t.test(df[df$dose == 'None', 'weeks_alive'], # Samples with no dose
- df[df$dose != 'None','weeks_alive'] # Samples with any dose
+
+
+PC3
+ANOVA! Remember that we have to call summary()
to see the ANOVA table:
+summary(aov(weeks_alive ~ dose, data = df))
+## Df Sum Sq Mean Sq F value Pr(>F)
+## dose 3 4052 1350.7 3.55 0.0245 *
+## Residuals 34 12937 380.5
+## ---
+## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
+This provides evidence in support of the alternative hypothesis that the group means are not equal. It seems that the dosage of Red Dye #40 has a relationship with how many weeks the mice lived.
+
+
+PC4
+T-test between None and Any, and between None and High.
+t.test(df$weeks_alive[df$dose == 'None'], # Samples with no dose
+ df$weeks_alive[df$dose != 'None'] # Samples with any dose
)
##
## Welch Two Sample t-test
##
-## data: df[df$dose == "None", "weeks_alive"] and df[df$dose != "None", "weeks_alive"]
+## data: df$weeks_alive[df$dose == "None"] and df$weeks_alive[df$dose != "None"]
## t = 4.2065, df = 33.732, p-value = 0.0001806
## alternative hypothesis: true difference in means is not equal to 0
## 95 percent confidence interval:
@@ -452,14 +376,13 @@ t.test(weeks_alive ~ dose == 'None', data = df)
## mean in group FALSE mean in group TRUE
## 69.11111 91.36364
# T-test between None and High
-
-t.test(df[df$dose == 'None', 'weeks_alive'], # Samples with no dose
- df[df$dose == 'High','weeks_alive'] # Samples with high dose
+t.test(df$weeks_alive[df$dose == 'None'], # Samples with no dose
+ df$weeks_alive[df$dose == 'High'] # Samples with high dose
)
##
## Welch Two Sample t-test
##
-## data: df[df$dose == "None", "weeks_alive"] and df[df$dose == "High", "weeks_alive"]
+## data: df$weeks_alive[df$dose == "None"] and df$weeks_alive[df$dose == "High"]
## t = 2.4958, df = 8.5799, p-value = 0.0353
## alternative hypothesis: true difference in means is not equal to 0
## 95 percent confidence interval:
@@ -467,85 +390,140 @@ t.test(df[df$dose == 'None', 'weeks_alive'], # Samples with no dose
## sample estimates:
## mean of x mean of y
## 91.36364 65.25000
-# Formula notation is a bit tricker. I would probably create a temprorary dataframe
-
-tmp = df %>% filter(dose %in% c('None', 'High'))
-
-t.test(weeks_alive ~ dose, data = tmp)
+# Formula notation is a bit tricker. I would create a temprorary dataframe
+df.tmp <- subset(df, subset= dose=="None" | dose=="High")
+t.test(weeks_alive ~ dose, data = df.tmp)
##
## Welch Two Sample t-test
##
## data: weeks_alive by dose
-## t = -2.4958, df = 8.5799, p-value = 0.0353
+## t = 2.4958, df = 8.5799, p-value = 0.0353
## alternative hypothesis: true difference in means is not equal to 0
## 95 percent confidence interval:
-## -49.960874 -2.266399
+## 2.266399 49.960874
## sample estimates:
-## mean in group High mean in group None
-## 65.25000 91.36364
-The t-test supports the idea that receiving a dose of RD40 reduces lifespan
-PC4. Anova
-summary(aov(weeks_alive ~ dose, data = df))
-## Df Sum Sq Mean Sq F value Pr(>F)
-## dose 3 4052 1350.7 3.55 0.0245 *
-## Residuals 34 12937 380.5
-## ---
-## Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
-This provides evidence that the group means are different.
+## mean in group None mean in group High
+## 91.36364 65.25000
+These t-tests both support the idea that receiving a dose of RD40 reduces lifespan. However, we should not completely trust these p-values, since we are doing multiple comparisons. One option is to do a Bonferroni correction, where we only cnsider things significant if \(\alpha < .05/m\) where \(m\) is the number of tests. In this case, we would fail to reject the null for the second test because we would set \(\alpha = .025\).
+The Bonferroni correction is more conservative than it needs to be, ane there are other approaches; for example, the TukeyHSD
function takes in an anova result and does post-hoc comparisons with corrections for all of the groups. The Reinhart book also provides a description of the Benjamini-Hochberg correction.
+
Q1. a) It is a sample statistic, because it comes from a sample. b) Confidence intervals for proportions are equal to
-\[p \pm z * \sqrt{ \frac{p*(1-p)}{n}}\]
-For a 95% confidence interval, \(z = 1.96\), so we can calculate it like this:
+It is a sample statistic (the sample mean), because it comes from a sample. The population parameter is unknown in this case, but can be estimated from the sample statistic.
As given in the textbook, confidence intervals for proportions are equal to:
\[\hat{p} \pm z^* \sqrt{ \frac{\hat{p}\times(1-\hat{p})}{n}}\]
+Where we calculate \(z^*\) from the Z distribution table. For a 95% confidence interval, \(z = 1.96\), so we can calculate it like this:
lower = .48 - 1.96 * sqrt(.48 * .52 / 1259)
upper = .48 + 1.96 * sqrt(.48 * .52 / 1259)
ci = c(lower, upper)
-
print(ci)
## [1] 0.4524028 0.5075972
This means that we are 95% confident that the true proportion of Americans who support legalizing marijuana is between ~45% and ~51%.
We have a large enough sample (collected randomly) in which enough responses are drawn from each potential outcome to assume that the observations are independent and the distribution of the sample proportion is approximately normal.
The statement is not justified, since our confidence interval includes 50%.
6.20 We can use the point estimate of the poll to estimate how large a sample we would need to have a confidence interval of a given width.
-Basically, we want each half of the confidence interval to be 1%, i.e., \(1.96 * \sqrt{\frac{.48 * .52}{n}} = .01\)
+We can use the point estimate of the poll to estimate how large a sample we would need to have a confidence interval of a given width.
+In this case, we want the margin or error to be 2%. Using the same formula from above, this translates to: \(1.96 * \sqrt{\frac{.48 * .52}{n}} = .02\)
We can solve for \(n\):
-\[\sqrt{\frac{.48 * .52}{n}} = .01/1.96\] \[\frac{.48 * .52}{n} = (.01/1.96)^2\] \[n = (.48 * .52)/(.01/1.96)^2\]
-(.48 * .52)/(.01/1.96)^2
-## [1] 9588.634
-So, we need a sample of approximately 9,589
-6.38
-The question is whether there has been a change in the proportion over time. While the tools we have learned could allow you to answer that question, they assume that responses are independent. In this case, they are obviously not independent as they come from the same students.
-6.50
+\[\sqrt{\frac{.48 * .52}{n}} = .02/1.96\] \[\frac{.48 * .52}{n} = (.02/1.96)^2\] \[n = (.48 * .52)/(.02/1.96)^2\] Let R solve this:
+(.48 * .52)/(.02/1.96)^2
+## [1] 2397.158
+So, we need a sample of at least 2,398 (since we canât survey less than a whole person).
+The question is whether there has been a change in the proportion across two groups (the students pre- and post-class) over time. While the tools we have learned could allow you to answer that question for two groups, they assume that responses are independent. In this case, the responses are not independent because they come from the same students. You could go ahead and calculate a statistical test for difference in pooled proportions (itâs just math!), but the dependence between observations violates the assumptions and means that the baseline expectations necessary to identify the hypothesis test are not met. Your test statistic will not mean what you want it to mean.
+chisq.test(x = c(83,121,193,103), p = c(.18,.22,.37,.23))
+Before we can calculate the test statistic we should check the conditions:
+1. Independence: \(500 < 10%\) of US population, and the sample is random, hence we can assume that one individual in the sample is independent of another.
+2. Sample size: The expected counts can be calculated as follows: \[E N E = 500 Ã 0.18 = 90\] \[E N C = 500 Ã 0.22 = 110\] \[E S = 500 Ã 0.37 = 185\] \[E W = 500 Ã 0.23 = 115\]
All expected counts are greater than 5 (which meets the rule of thumb re: empty âcellsâ in the table).
+We can test this with a \(\chi^2\) test.
+# Manually using the expected counts and formula for chi-squared
+chisq = (83-90)^2/90 + (121-110)^2/110 + (193-185)^2/185 + (103-115)^2/115
+chisq
+## [1] 3.242564
+# We could then look up the chi-square distribution for 3 degrees of freedom in the book or find it in R.
+# lower.tail=FALSE ensures that R returns the proportion of the distribution at least as large as our critical value:
+pchisq(chisq, df=3, lower.tail=FALSE)
+## [1] 0.355717
+# We could also use R's chisq.test function:
+chisq.test(x = c(83,121,193,103), p = c(.18,.22,.37,.23))
##
## Chi-squared test for given probabilities
##
## data: c(83, 121, 193, 103)
## X-squared = 3.2426, df = 3, p-value = 0.3557
-# Or manually
-
-# Calculate expected values
-500 * c(.18,.22,.37,.23)
-## [1] 90 110 185 115
-# Use the formula for chi-squared
-chisq = (83-90)^2/90 + (121-110)^2/110 + (193-185)^2/185 + (103-115)^2/115
-The p-value for this is large, meaning that we donât have evidence that the sample differs from the census distribution.
+The p-value for this is large, meaning that we cannot reject the null hypothesis that the sample proportions do not follow the census distribution.
You can do this by hand calculating the expected counts and then plugging the values into the formula for \(\chi^2\) as we did above. If you do that, here are the expected counts:
+Right direction:
+# NE:
+round(171*83/500)
+## [1] 28
+# NC:
+round(171*121/500)
+## [1] 41
+# S:
+round(171*193/500)
+## [1] 66
+# W:
+round(171*103/500)
+## [1] 35
+Wrong direction:
+round(329*83/500)
+## [1] 55
+round(329*121/500)
+## [1] 80
+round(329*193/500)
+## [1] 127
+round(329*103/500)
+## [1] 68
+Working through the formula for the \(\chi^2\) test statistic and then calculating the p-value for 3 degrees of freedom:
+chisq <- (29-28)^2 /28 +
+ (44-41)^2 /41 +
+ (62-66)^2 /66 +
+ (36-35)^2 / 35 +
+ (54-55)^2 / 55 +
+ (77-80)^2 / 80 +
+ (131-127)^2 / 127 +
+ (67-68)^2 / 68
+
+chisq
+## [1] 0.7975941
+pchisq(chisq, df=3, lower.tail=FALSE)
+## [1] 0.8500424
+And using Râs chisq.test()
function here:
x <- matrix(c(29,54,44,77,62,131,36,67), nrow = 4, # this makes a matrix with 4 rows
byrow=T) # And this says that we've entered it row by row
chisq.test(x)
@@ -554,6 +532,69 @@ chisq.test(x)
##
## data: x
## X-squared = 0.66724, df = 3, p-value = 0.8809
+The answers donât match exactly (because of the rounding I did in computing the expected counts), however the payoff/interpretation is the same: we cannot reject the null hypothesis of no relationship between region and direction opinoin in the sample.
+The unit of analysis is the customer. The dependent variable is the type of board purchased and the independent variable is gender. Males, females, and unkown gender customers are being compared. This is a two-way test.
For this type of comparison statistical tests help to give (or take away) confidence in any observed differences across categories. Choosing a statistical test is based on the question that you want to answer and the type of data that you have available to answer it. For example, if this were numeric data (e.g., the amount of money spent on electronics for men and women) then we could choose a t-test to compare those distributions.
The null hypothesis (\(H_0\)) is that the board purchased is independent of the gender of the customer. The alternative hypothesis (\(H_A\)) is that board purchase choice is dependent on gender.
A \(\chi^2\) test found statistically significant evidence that board purchase behavior differs by gender. This difference is convincing, but it does directly not tell us what the authors set out to understand, which is the difference between men and women (the test could have identified a significant difference in the number of unknown gender customers across board types). Many of these concerns are addressed in the text and with additional tests, giving increased confidence in the observed differences.
These are counts for two categorical variables, so the procedure used was a \(\chi^2\) test. The null hypothesis is that whether or not a blog is governed by one person is independent of whether it is on the left or the right ideologically.
Assuming that the null hypothesis of no difference across the two groups is compelling, it would be surprising to see these results in a world where ideological orientation and blog governance have no relationship. In this respect, it makes sense to believe that this difference is likely real. Perhaps the main reason to be skeptical is the way that the data are grouped. The authors could have grouped them differently (e.g., 1-2 people, 3-4 people, and 5+ people); if the decision on how to group was made after seeing the data then we have good reason to be skeptical.
We can do this in R.
# First we create the dataframe
+df = data.frame(Governance=c('Individual','Multiple', 'Individual','Multiple'),
+ Ideology = c('Left','Left','Right','Right'),
+ Count = c(13,51,27,38))
+
+# We can make sure it's the same by testing the Chi-squared
+chisq.test(matrix(df$Count, nrow=2))
+##
+## Pearson's Chi-squared test with Yates' continuity correction
+##
+## data: matrix(df$Count, nrow = 2)
+## X-squared = 5.8356, df = 1, p-value = 0.01571
+# Using Jeremy's tidyverse code:
+percentage_data <- df %>%
+ group_by(Ideology) %>%
+ summarize(individual_ratio = sum(Count[Governance=='Individual']) / sum(Count),
+ group_count = sum(Count))
+
+shaw_benkler_plot = percentage_data %>%
+ ggplot(aes(x=Ideology, y=individual_ratio * 100)) +
+ geom_bar(stat='identity', aes(fill=c('red','blue')), show.legend=F) +
+ ylab('Percentage of Blogs') + theme_minimal()
+
+shaw_benkler_plot
+
+If we want to add error bars, we need to calculate them (Note that ggplot could do this for us if we had raw data - always share your data!).
+For our purposes here, Jeremy decided to use confidence intervals. The standard error is another reasonable choice. Either way, ggplot has a geom_errorbar
layer that is very useful.
Remember that for a binomial distribution (we can consider individual/non-individual as binomial), confidence intervals are \(\hat{p} \pm z^* \sqrt{\frac{\hat{p}~(1-\hat{p})}{n}}\)
+ci_95 = 1.96 * sqrt(percentage_data$individual_ratio * (1 - percentage_data$individual_ratio)/percentage_data$group_count)
+
+shaw_benkler_plot + geom_errorbar(aes(ymin=(individual_ratio-ci_95)*100, ymax=(individual_ratio + ci_95)*100),
+ alpha = .3,
+ size=1.1,
+ width=.4)
+
+The error bars do overlap in this case, indicating that the true population proportions may not be as far apart as our point estimates suggest. Note that this is not the same as the hypothesis test.
+Another way in which the base rate fallacy could play a role in this paper, however, concerns the presence of multiple comparisons. The authors conducted numerous statistical tests (indeed, one of the authors seems to recall that some of the tests were not even reported in the paper
In any case, the point here is that the statistical tests reported in the paper may not mean exactly what the authors said they did in the context of the publication. That may or may not change the validity of the results, but it should inspire us all to do better statistical analysis!
+