This repository has been archived by the owner on Dec 28, 2023. It is now read-only.
forked from moderndive/ModernDive_book
-
Notifications
You must be signed in to change notification settings - Fork 13
/
92-appendixB.Rmd
executable file
·1355 lines (934 loc) · 58.8 KB
/
92-appendixB.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
# Inference Examples {#appendixB}
This appendix is designed to provide you with examples of the five basic hypothesis tests and their corresponding confidence intervals. Traditional theory-based methods as well as computational-based methods are presented. You can also use this appendix as a way to check for understanding of which statistical graphic is most appropriate given the problem set-up.
```{r include=FALSE}
knitr::opts_chunk$set(tidy = FALSE,
out.width = '\\textwidth',
warning = FALSE,
message = FALSE)
```
## Needed packages {-}
```{r}
library(dplyr)
library(ggplot2)
library(mosaic)
library(knitr)
library(readr)
```
## Inference mind map
To help you better navigate and choose the appropriate analysis, we've created a mind map on <http://coggle.it> available [here](https://coggle.it/diagram/Vxlydu1akQFeqo6-) and below.
```{r infer-map, echo=FALSE, fig.cap="Mind map for Inference", out.width="200%"}
#library(knitr)
#if(knitr:::is_html_output()){
# include_url("https://coggle.it/diagram/Vxlydu1akQFeqo6-", height="1200px")
#} else {
include_graphics("images/coggle.png")
#library(magick)
#image_read("images/coggle.png") %>% image_scale("300")
#}
```
## One mean
### Problem statement
The National Survey of Family Growth conducted by the
Centers for Disease Control gathers information on family life, marriage and divorce, pregnancy,
infertility, use of contraception, and men's and women's health. One of the variables collected on
this survey is the age at first marriage. 5,534 randomly sampled US women between 2006 and 2010 completed the survey. The women sampled here had been married at least once. Do we have evidence that the mean age of first marriage for all US women from 2006 to 2010 is greater than 23 years? [Tweaked a bit from @isrs2014 [Chapter 4]]
### Competing hypotheses
#### In words
- Null hypothesis: The mean age of first marriage for all US women from 2006 to 2010 is equal to 23 years.
- Alternative hypothesis: The mean age of first marriage for all US women from 2006 to 2010 is greater than 23 years.
#### In symbols (with annotations)
- $H_0: \mu = \mu_{0}$, where $\mu$ represents the mean age of first marriage for all US women from 2006 to 2010 and $\mu_0$ is 23.
- $H_A: \mu > 23$
#### Set $\alpha$
It's important to set the significance level before starting the testing using the data. Let's set the significance level at 5\% here.
### Exploring the sample data
```{r read_data1}
#download.file("http://ismayc.github.io/teaching/sample_problems/ageAtMar.csv",
# destfile = "data/ageAtMar.csv",
# method = "curl")
ageAtMar <- read_csv("data/ageAtMar.csv")
```
```{r summarize1}
age_summ <- ageAtMar %>%
summarize(sample_size = n(),
mean = mean(age),
sd = sd(age),
minimum = min(age),
lower_quartile = quantile(age, 0.25),
median = median(age),
upper_quartile = quantile(age, 0.75),
max = max(age))
kable(age_summ)
```
The histogram below also shows the distribution of `age`.
```{r hist1b}
ageAtMar %>% ggplot(aes(x = age)) +
geom_histogram(binwidth = 3, color = "white")
```
#### Guess about statistical significance
We are looking to see if the observed sample mean of `r age_summ$mean` is statistically greater than $\mu_0 = 23$. They seem to be quite close, but we have a large sample size here. Let's guess that the large sample size will lead us to reject this practically small difference.
***
### Non-traditional methods
```{r include=FALSE}
mu0 <- 23
```
#### Bootstrapping for hypothesis test
In order to look to see if the observed sample mean of `r age_summ$mean` is statistically greater than $\mu_0 = 23$, we need to account for the sample size. We also need to determine a process that replicates how the original sample of size `r nrow(ageAtMar)` was selected.
We can use the idea of *bootstrapping* to simulate the population from which the sample came and then generate samples from that simulated population to account for sampling variability. Recall how bootstrapping would apply in this context:
1. Sample with replacement from our original sample of `r nrow(ageAtMar)` women and repeat this process 10,000 times,
2. calculate the mean for each of the 10,000 bootstrap samples created in Step 1.,
3. combine all of these bootstrap statistics calculated in Step 2 into a `boot_distn` object, and
4. shift the center of this distribution over to the null value of `r mu0`. (This is needed since it will be centered at `r age_summ$mean` via the process of bootstrapping.)
```{r, echo=FALSE}
if(!file.exists("rds/null_distn_one_mean.rds")){
set.seed(2017)
mu0 <- 23
shift <- mu0 - age_summ$mean
null_distn_one_mean <- do(10000) *
resample(ageAtMar, replace = TRUE) %>%
mutate(age = age + shift) %>%
summarize(mean_age = mean(age))
saveRDS(object = null_distn_one_mean, "rds/null_distn_one_mean.rds")
} else {
null_distn_one_mean <- readRDS("rds/null_distn_one_mean.rds")
}
```
```{r sim1, eval=FALSE}
set.seed(2017)
mu0 <- 23
shift <- mu0 - age_summ$mean
null_distn_one_mean <- do(10000) *
resample(ageAtMar, replace = TRUE) %>%
mutate(age = age + shift) %>%
summarize(mean_age = mean(age))
```
```{r}
ggplot(null_distn_one_mean, aes(x = mean_age)) +
geom_histogram(bins = 30, color = "white")
```
We can next use this distribution to observe our $p$-value. Recall this is a right-tailed test so we will be looking for values that are greater than or equal to `r age_summ$mean` for our $p$-value.
```{r}
obs_mean <- age_summ$mean
ggplot(null_distn_one_mean, aes(x = mean_age)) +
geom_histogram(bins = 30, color = "white") +
geom_vline(color = "red", xintercept = obs_mean)
```
##### Calculate $p$-value
```{r}
pvalue <- null_distn_one_mean %>%
filter( mean_age >= obs_mean ) %>%
nrow() / nrow(null_distn_one_mean)
pvalue
```
So our $p$-value is `r pvalue` and we reject the null hypothesis at the 5% level. You can also see this from the histogram above that we are far into the tail of the null distribution.
#### Bootstrapping for confidence interval
We can also create a confidence interval for the unknown population parameter $\mu$ using our sample data using *bootstrapping*. Note that we don't need to shift this distribution since we want the center of our confidence interval to be our point estimate $\bar{x}_{obs} = `r obs_mean`$.
```{r, echo=FALSE}
if(!file.exists("rds/boot_distn_one_mean.rds")){
boot_distn_one_mean <- do(10000) *
resample(ageAtMar, replace = TRUE) %>%
summarize(mean_age = mean(age))
saveRDS(object = boot_distn_one_mean, "rds/boot_distn_one_mean.rds")
} else {
boot_distn_one_mean <- readRDS("rds/boot_distn_one_mean.rds")
}
```
```{r boot1, eval=FALSE}
boot_distn_one_mean <- do(10000) *
resample(ageAtMar, replace = TRUE) %>%
summarize(mean_age = mean(age))
```
```{r}
ggplot(boot_distn_one_mean, aes(x = mean_age)) +
geom_histogram(bins = 30, color = "white")
```
```{r}
boot_distn_one_mean %>% summarize(lower = quantile(mean_age, probs = 0.025),
upper = quantile(mean_age, probs = 0.975))
```
We see that `r mu0` is not contained in this confidence interval as a plausible value of $\mu$ (the unknown population mean) and the entire interval is larger than `r mu0`. This matches with our hypothesis test results of rejecting the null hypothesis in favor of the alternative ($\mu > 23$).
**Interpretation**: We are 95% confident the true mean age of first marriage for all US women from 2006 to 2010 is between `r boot_distn_one_mean$lower` and `r boot_distn_one_mean$upper`.
---
### Traditional methods
#### Check conditions
Remember that in order to use the shortcut (formula-based, theoretical) approach, we need to check that some conditions are met.
1. _Independent observations_: The observations are collected independently.
The cases are selected independently through random sampling so this condition is met.
2. _Approximately normal_: The distribution of the response variable should be normal or the sample size should be at least 30.
The histogram for the sample above does show some skew.
The Q-Q plot below also shows some skew.
```{r qqplotmean}
ggplot(data = ageAtMar, mapping = aes(sample = age)) +
stat_qq()
```
The sample size here is quite large though ($n = 5534$) so both conditions are met.
#### Test statistic
The test statistic is a random variable based on the sample data. Here, we want to look at a way to estimate the population mean $\mu$. A good guess is the sample mean $\bar{X}$. Recall that this sample mean is actually a random variable that will vary as different samples are (theoretically, would be) collected. We are looking to see how likely is it for us to have observed a sample mean of $\bar{x}_{obs} = `r obs_mean`$ or larger assuming that the population mean is `r mu0` (assuming the null hypothesis is true). If the conditions are met and assuming $H_0$ is true, we can "standardize" this original test statistic of $\bar{X}$ into a $T$ statistic that follows a $t$ distribution with degrees of freedom equal to $df = n - 1$:
$$ T =\dfrac{ \bar{X} - \mu_0}{ S / \sqrt{n} } \sim t (df = n - 1) $$
where $S$ represents the standard deviation of the sample and $n$ is the sample size.
##### Observed test statistic
While one could compute this observed test statistic by "hand", the focus here is on the set-up of the problem and in understanding which formula for the test statistic applies. We can use the `t.test` function to perform this analysis for us.
```{r t.test1a}
t.test(x = ageAtMar$age,
alternative = "greater",
mu = 23)
```
```{r inference1, eval=FALSE, include=FALSE}
inference(y = ageAtMar$age,
est = "mean",
null = 23,
alternative = "greater",
type = "ht",
method = "theoretical",
eda_plot = FALSE,
inf_plot = FALSE)
```
We see here that the $t_{obs}$ value is around 6.94. Recall that for large sample sizes the $t$ distribution is essentially the standard normal distribution and this is why the statistic is reported as `Z`.
#### Compute $p$-value
The $p$-value---the probability of observing an $t_{obs}$ value of 6.94 or more in our null distribution of a $t$ with 5433 degrees of freedom---is essentially 0. This can also be calculated in R directly:
```{r pval1}
pt(6.936, df = nrow(ageAtMar) - 1, lower.tail = FALSE)
```
We can also use the $N(0, 1)$ distribution here:
```{r pval2}
pnorm(6.936, lower.tail = FALSE)
```
#### State conclusion
We, therefore, have sufficient evidence to reject the null hypothesis. Our initial guess that our observed sample mean was statistically greater than the hypothesized mean has supporting evidence here. Based on this sample, we have evidence that the mean age of first marriage for all US women from 2006 to 2010 is greater than 23 years.
#### Confidence interval
The confidence interval reported above with `t.test` is known as a one-sided confidence interval and gives the lowest value one could expect $\mu$ to be with 95% confidence. We usually want a range of values so we can use `alternative = "two.sided"` to get the similar values compared to the bootstrapping process:
```{r t.test1}
t.test(x = ageAtMar$age,
alternative = "two.sided",
mu = 23)$conf
```
***
### Comparing results
Observing the bootstrap distribution that were created, it makes quite a bit of sense that the results are so similar for traditional and non-traditional methods in terms of the $p$-value and the confidence interval since these distributions look very similar to normal distributions. The conditions also being met (the large sample size was the driver here) leads us to better guess that using any of the methods whether they are traditional (formula-based) or non-traditional (computational-based) will lead to similar results.
***
***
## One proportion
### Problem statement
The CEO of a large electric utility claims that 80 percent of his 1,000,000 customers are satisfied with the service they receive. To test this claim, the local newspaper surveyed 100 customers, using simple random sampling. 73 were satisfied and the remaining were unsatisfied. Based on these findings from the sample, can we reject the CEO's hypothesis that 80% of the customers are satisfied? [Tweaked a bit from http://stattrek.com/hypothesis-test/proportion.aspx?Tutorial=AP]
### Competing hypotheses
#### In words
- Null hypothesis: The proportion of all customers of the large electric utility satisfied with service they receive is equal 0.80.
- Alternative hypothesis: The proportion of all customers of the large electric utility satisfied with service they receive is different from 0.80.
<!--
## Another way with words
- Null hypothesis: There is no association between feed type and age in the population of interest.
- Alternative hypothesis: There is an association between the variables in the population.
-->
#### In symbols (with annotations)
- $H_0: \pi = p_{0}$, where $\pi$ represents the proportion of all customers of the large electric utility satisfied with service they receive and $p_0$ is 0.8.
- $H_A: \pi \ne 0.8$
#### Set $\alpha$
It's important to set the significance level before starting the testing using the data. Let's set the significance level at 5% here.
### Exploring the sample data
```{r read_data2}
elec <- c(rep("satisfied", 73), rep("unsatisfied", 27)) %>%
as_data_frame() %>%
rename("satisfy" = value)
```
The bar graph below also shows the distribution of `satisfy`.
```{r bar}
ggplot(data = elec, aes(x = satisfy)) +
geom_bar()
```
#### Guess about statistical significance
We are looking to see if the sample proportion of `r prop.table(table(elec$satisfy))[1]` is statistically different from $p_0 = 0.8$ based on this sample. They seem to be quite close, and our sample size is not huge here ($n = 100$). Let's guess that we do not have evidence to reject the null hypothesis.
***
### Non-traditional methods
#### Simulation for hypothesis test
In order to look to see if `r prop.table(table(elec$satisfy))[1]` is statistically different from 0.8, we need to account for the sample size. We also need to determine a process that replicates how the original sample of size 100 was selected. We can use the idea of an unfair coin to *simulate* this process. We will simulate flipping an unfair coin (with probability of success 0.8 matching the null hypothesis) 100 times. Then we will keep track of how many heads come up in those 100 flips. Our simulated statistic matches with how we calculated the original statistic $\hat{p}$: the number of heads (satisfied) out of our total sample of 100. We then repeat this process many times (say 10,000) to create the null distribution looking at the simulated proportions of successes:
```{r, echo=FALSE}
if(!file.exists("rds/null_distn_one_prop.rds")){
set.seed(2017)
null_distn_one_prop <- do(10000) * rflip(100, prob = 0.8)
saveRDS(object = null_distn_one_prop, "rds/null_distn_one_prop.rds")
} else {
null_distn_one_prop <- readRDS("rds/null_distn_one_prop.rds")
}
```
```{r sim2, eval=FALSE}
set.seed(2017)
null_distn_one_prop <- do(10000) * rflip(100, prob = 0.8)
```
```{r}
ggplot(null_distn_one_prop, aes(x = prop)) +
geom_histogram(bins = 30, color = "white")
```
We can next use this distribution to observe our $p$-value. Recall this is a two-tailed test so we will be looking for values that are 0.8 - 0.73 = 0.07 away from 0.8 in BOTH directions for our $p$-value:
```{r}
p_hat <- 73/100
dist <- 0.8 - p_hat
ggplot(null_distn_one_prop, aes(x = prop)) +
geom_histogram(bins = 30, color = "white") +
geom_vline(color = "red", xintercept = 0.8 + dist) +
geom_vline(color = "red", xintercept = p_hat)
```
##### Calculate $p$-value
```{r}
pvalue <- null_distn_one_prop %>%
filter( (prop >= 0.8 + dist) | (prop <= p_hat) ) %>%
nrow() / nrow(null_distn_one_prop)
pvalue
```
So our $p$-value is `r pvalue` and we fail to reject the null hypothesis at the 5% level.
#### Bootstrapping for confidence interval
We can also create a confidence interval for the unknown population parameter $\pi$ using our sample data. To do so, we use *bootstrapping*, which involves
1. sampling with replacement from our original sample of 100 survey respondents and repeating this process 10,000 times,
2. calculating the proportion of successes for each of the 10,000 bootstrap samples created in Step 1.,
3. combining all of these bootstrap statistics calculated in Step 2 into a `boot_distn` object,
4. identifying the 2.5^th^ and 97.5^th^ percentiles of this distribution (corresponding to the 5% significance level chosen) to find a 95% confidence interval for $\pi$, and
5. interpret this confidence interval in the context of the problem.
```{r, echo=FALSE}
if(!file.exists("rds/boot_distn_one_prop.rds")){
boot_distn_one_prop <- do(10000) *
(elec %>% resample(size = 100, replace = TRUE) )%>%
summarize(success_rate = mean(satisfy == "satisfied"))
saveRDS(object = boot_distn_one_prop, "rds/boot_distn_one_prop.rds")
} else {
boot_distn_one_prop <- readRDS("rds/boot_distn_one_prop.rds")
}
```
```{r boot2, eval=FALSE}
boot_distn_one_prop <- do(10000) *
(elec %>% resample(size = 100, replace = TRUE) )%>%
summarize(success_rate = mean(satisfy == "satisfied"))
```
Just as we use the `mean` function for calculating the mean over a numerical variable, we can also use it to compute the proportion of successes for a categorical variable where we specify what we are calling a "success" after the `==`. (Think about the formula for calculating a mean and how R handles logical statements such as `satisfy == "satisfied"` for why this must be true.)
```{r}
ggplot(boot_distn_one_prop, aes(x = success_rate)) +
geom_histogram(bins = 30, color = "white")
```
```{r}
boot_distn_one_prop %>% summarize(lower = quantile(success_rate, probs = 0.025),
upper = quantile(success_rate, probs = 0.975))
```
We see that 0.80 is contained in this confidence interval as a plausible value of $\pi$ (the unknown population proportion). This matches with our hypothesis test results of failing to reject the null hypothesis.
**Interpretation**: We are 95% confident the true proportion of customers who are satisfied with the service they receive is between `r boot_distn_one_prop$lower` and `r boot_distn_one_prop$upper`.
**Note**: You could also use the null distribution with a shift to have its center at $\hat{p} = 0.73$ instead of at $p_0 = 0.8$ and calculate its percentiles. The confidence interval produced via this method should be comparable to the one done using bootstrapping above.
---
### Traditional methods
#### Check conditions
Remember that in order to use the shortcut (formula-based, theoretical) approach, we need to check that some conditions are met.
1. _Independent observations_: The observations are collected independently.
The cases are selected independently through random sampling so this condition is met.
2. _Approximately normal_: The number of expected successes and expected failures is at least 10.
This condition is met since 73 and 27 are both greater than 10.
#### Test statistic
The test statistic is a random variable based on the sample data. Here, we want to look at a way to estimate the population proportion $\pi$. A good guess is the sample proportion $\hat{P}$. Recall that this sample proportion is actually a random variable that will vary as different samples are (theoretically, would be) collected. We are looking to see how likely is it for us to have observed a sample proportion of $\hat{p}_{obs} = `r p_hat`$ or larger assuming that the population proportion is 0.80 (assuming the null hypothesis is true). If the conditions are met and assuming $H_0$ is true, we can standardize this original test statistic of $\hat{P}$ into a $Z$ statistic that follows a $N(0, 1)$ distribution.
$$ Z =\dfrac{ \hat{P} - p_0}{\sqrt{\dfrac{p_0(1 - p_0)}{n} }} \sim N(0, 1) $$
##### Observed test statistic
While one could compute this observed test statistic by "hand" by plugging the observed values into the formula, the focus here is on the set-up of the problem and in understanding which formula for the test statistic applies. The calculation has been done in R below for completeness though:
```{r}
p_hat <- 0.73
p0 <- 0.8
n <- 100
(z_obs <- (p_hat - p0) / sqrt( (p0 * (1 - p0)) / n))
```
We see here that the $z_{obs}$ value is around `r z_obs`. Our observed sample proportion of `r p_hat` is `r -z_obs` standard errors below the hypothesized parameter value of `r p0`.
#### Compute $p$-value
```{r pvaloneprop}
2 * pnorm(z_obs)
```
The $p$-value---the probability of observing an $z_{obs}$ value of `r z_obs` or more extreme (in both directions) in our null distribution---is around `r round(2 * pnorm(z_obs) * 100)`%.
Note that we could also do this test directly using the `prop.test` function.
```{r prop.test1}
stats::prop.test(x = table(elec$satisfy),
n = length(elec$satisfy),
alternative = "two.sided",
p = 0.8,
correct = FALSE)
```
`prop.test` does a $\chi^2$ test here but this matches up exactly with what we would expect: $x^2_{obs} = 3.06 = (-1.75)^2 = (z_{obs})^2$ and the $p$-values are the same because we are focusing on a two-tailed test.
Note that the 95 percent confidence interval given above matches well with the one calculated using bootstrapping.
#### State conclusion
We, therefore, do not have sufficient evidence to reject the null hypothesis. Our initial guess that our observed sample proportion was not statistically greater than the hypothesized proportion has not been invalidated. Based on this sample, we have do not evidence that the proportion of all customers of the large electric utility satisfied with service they receive is different from 0.80, at the 5% level.
***
### Comparing results
Observing the bootstrap distribution and the null distribution that were created, it makes quite a bit of sense that the results are so similar for traditional and non-traditional methods in terms of the $p$-value and the confidence interval since these distributions look very similar to normal distributions. The conditions also being met leads us to better guess that using any of the methods whether they are traditional (formula-based) or non-traditional (computational-based) will lead to similar results.
***
***
## Two proportions
### Problem statement
A 2010 survey asked 827 randomly sampled registered voters
in California "Do you support? Or do you oppose? Drilling for oil and natural gas off the Coast of
California? Or do you not know enough to say?" Conduct a hypothesis test to determine if the data
provide strong evidence that the proportion of college
graduates who do not have an opinion on this issue is
different than that of non-college graduates. [Tweaked a bit from @isrs2014 [Chapter 6]]
### Competing hypotheses
#### In words
- Null hypothesis: There is no association between having an opinion on drilling and having a college degree for all registered California voters in 2010.
- Alternative hypothesis: There is an association between having an opinion on drilling and having a college degree for all registered California voters in 2010.
#### Another way in words
- Null hypothesis: The probability that a Californian voter in 2010 having no opinion on drilling and is a college graduate is the **same** as that of a non-college graduate.
- Alternative hypothesis: These parameter probabilities are different.
#### In symbols (with annotations)
- $H_0: \pi_{college} = \pi_{no\_college}$ or $H_0: \pi_{college} - \pi_{no\_college} = 0$, where $\pi$ represents the probability of not having an opinion on drilling.
- $H_A: \pi_{college} - \pi_{no\_college} \ne 0$
#### Set $\alpha$
It's important to set the significance level before starting the testing using the data. Let's set the significance level at 5% here.
### Exploring the sample data
```{r read_data3, warning=FALSE, message=FALSE}
#download.file("http://ismayc.github.io/teaching/sample_problems/offshore.csv",
# destfile = "data/offshore.csv",
# method = "curl")
offshore <- read_csv("data/offshore.csv")
```
```{r summarize2}
table(offshore$college_grad, offshore$response)
off_summ <- offshore %>%
group_by(college_grad) %>%
summarize(prop_no_opinion = mean(response == "no opinion"),
sample_size = n())
```
```{r stacked_bar}
ggplot(offshore, aes(x = college_grad, fill = response)) +
geom_bar(position = "fill") +
coord_flip()
```
#### Guess about statistical significance
We are looking to see if a difference exists in the heights of the bars corresponding to `no opinion` for the plot. Based solely on the plot, we have little reason to believe that a difference exists since the bars seem to be about the same height, BUT...it's important to use statistics to see if that difference is actually statistically significant!
***
### Non-traditional methods
#### Collecting summary info
Next we will assign some key values to variable names in R:
```{r stats1}
phat_nograd <- off_summ$prop_no_opinion[1]
phat_grad <- off_summ$prop_no_opinion[2]
obs_diff <- phat_grad - phat_nograd
n_nograd <- off_summ$sample_size[1]
n_grad <- off_summ$sample_size[2]
```
#### Randomization for hypothesis test
In order to look to see if the observed sample proportion of no opinion for college graduates of `r phat_nograd` is statistically different than that for graduates of `r phat_grad`, we need to account for the sample sizes. Note that this is the same as looking to see if $\hat{p}_{grad} - \hat{p}_{nograd}$ is statistically different than 0. We also need to determine a process that replicates how the original group sizes of `r n_nograd` and `r n_grad` were selected.
We can use the idea of *randomization testing* (also known as *permutation testing*) to simulate the population from which the sample came (with two groups of different sizes) and then generate samples using *shuffling* from that simulated population to account for sampling variability.
```{r, echo=FALSE}
if(!file.exists("rds/null_distn_two_props.rds")){
set.seed(2017)
many_shuffles <- do(10000) *
(offshore %>%
mutate(college_grad = shuffle(college_grad)) %>%
group_by(college_grad) %>%
summarize(prop_no_opinion = mean(response == "no opinion"))
)
null_distn_two_props <- many_shuffles %>%
group_by(.index) %>%
summarize(diffprop = diff(prop_no_opinion))
saveRDS(object = null_distn_two_props, "rds/null_distn_two_props.rds")
} else {
null_distn_two_props <- readRDS("rds/null_distn_two_props.rds")
}
```
```{r sim3, eval=FALSE}
set.seed(2017)
many_shuffles <- do(10000) *
(offshore %>%
mutate(college_grad = shuffle(college_grad)) %>%
group_by(college_grad) %>%
summarize(prop_no_opinion = mean(response == "no opinion"))
)
null_distn_two_props <- many_shuffles %>%
group_by(.index) %>%
summarize(diffprop = diff(prop_no_opinion))
```
```{r}
ggplot(null_distn_two_props, aes(x = diffprop)) +
geom_histogram(bins = 25, color = "white")
```
We can next use this distribution to observe our $p$-value. Recall this is a two-tailed test so we will be looking for values that are greater than or equal to `r obs_diff` or less than or equal to `r -obs_diff` for our $p$-value.
```{r}
ggplot(null_distn_two_props, aes(x = diffprop)) +
geom_histogram(bins = 20, color = "white") +
geom_vline(color = "red", xintercept = obs_diff) +
geom_vline(color = "red", xintercept = -obs_diff)
```
##### Calculate $p$-value
```{r}
pvalue <- null_distn_two_props %>%
filter( (diffprop <= obs_diff) | (diffprop >= -obs_diff) ) %>%
nrow() / nrow(null_distn_two_props)
pvalue
```
So our $p$-value is `r pvalue` and we reject the null hypothesis at the 5% level. You can also see this from the histogram above that we are far into the tails of the null distribution.
#### Bootstrapping for confidence interval
We can also create a confidence interval for the unknown population parameter $\pi_{college} - \pi_{no\_college}$ using our sample data with *bootstrapping*. Here we will bootstrap each of the groups with replacement instead of shuffling. This is done using the `groups`
argument in the `resample` function to fix the size of each group to
be the same as the original group sizes of `r n_nograd` for non-college graduates and `r n_grad` for college graduates.
```{r, echo=FALSE}
if(!file.exists("rds/boot_distn_two_props.rds")){
boot_props <- do(10000) *
offshore %>%
resample(replace = TRUE, groups = college_grad) %>%
group_by(college_grad) %>%
summarize(prop_no_opinion = mean(response == "no opinion"))
# Next, we calculate the difference in sample proportions for each of the 10,000 replications:
boot_distn_two_props <- boot_props %>%
group_by(.index) %>%
summarize(diffprop = diff(prop_no_opinion))
saveRDS(object = boot_distn_two_props, "rds/boot_distn_two_props.rds")
} else {
boot_distn_two_props <- readRDS("rds/boot_distn_two_props.rds")
}
```
```{r boot3, eval=FALSE}
boot_props <- do(10000) *
offshore %>%
resample(replace = TRUE, groups = college_grad) %>%
group_by(college_grad) %>%
summarize(prop_no_opinion = mean(response == "no opinion"))
# Next, we calculate the difference in sample proportions for each of the 10,000 replications:
boot_distn_two_props <- boot_props %>%
group_by(.index) %>%
summarize(diffprop = diff(prop_no_opinion))
```
```{r}
ggplot(boot_distn_two_props, aes(x = diffprop)) +
geom_histogram(bins = 30, color = "white")
```
```{r}
ci_boot <- boot_distn_two_props %>%
summarize(lower = quantile(diffprop, probs = 0.025),
upper = quantile(diffprop, probs = 0.975))
ci_boot
```
We see that 0 is not contained in this confidence interval as a plausible value of $\pi_{college} - \pi_{no\_college}$ (the unknown population parameter). This matches with our hypothesis test results of rejecting the null hypothesis. Since zero is not a plausible value of the population parameter, we have evidence that the proportion of college graduates in California with no opinion on drilling is different than that of non-college graduates.
**Interpretation**: We are 95% confident the true proportion of non-college graduates with no opinion on offshore drilling in California is between `r round(-ci_boot$lower, 2)` dollars smaller to `r round(-ci_boot$upper, 2)` dollars smaller than for college graduates.
**Note**: You could also use the null distribution based on randomization with a shift to have its center at $\hat{p}_{college} - \hat{p}_{no\_college} = \$`r round(obs_diff, 2)`$ instead of at 0 and calculate its percentiles. The confidence interval produced via this method should be comparable to the one done using bootstrapping above.
---
### Traditional methods
### Check conditions
Remember that in order to use the short-cut (formula-based, theoretical) approach, we need to check that some conditions are met.
1. _Independent observations_: Each case that was selected must be independent of all the other cases selected.
This condition is met since cases were selected at random to observe.
2. _Sample size_: The number of pooled successes and pooled failures must be at least 10 for each group.
We need to first figure out the pooled success rate: $$\hat{p}_{obs} = \dfrac{131 + 104}{827} = 0.28.$$ We now determine expected (pooled) success and failure counts:
$0.28 \cdot (131 + 258) = `r 0.28 * (131 + 258)`$, $0.72 \cdot (131 + 258) = `r 0.72 * (131 + 258)`$
$0.28 \cdot (104 + 334) = `r 0.28 * (104 + 334)`$, $0.72 \cdot (104 + 334) = `r 0.72 * (104 + 334)`$
3. _Independent selection of samples_: The cases are not paired in any meaningful way.
We have no reason to suspect that a college graduate selected would have any relationship to a non-college graduate selected.
### Test statistic
The test statistic is a random variable based on the sample data. Here, we are interested in seeing if our observed difference in sample proportions corresponding to no opinion on drilling ($\hat{p}_{college, obs} - \hat{p}_{no\_college, obs}$ = `r prop.table(table(offshore$college_grad, offshore$response))[1, 1] - prop.table(table(offshore$college_grad, offshore$response))[2, 1]`) is statistically different than 0. Assuming that conditions are met and the null hypothesis is true, we can use the standard normal distribution to standardize the difference in sample proportions ($\hat{P}_{college} - \hat{P}_{no\_college}$) using the standard error of $\hat{P}_{college} - \hat{P}_{no\_college}$ and the pooled estimate:
$$ Z =\dfrac{ (\hat{P}_1 - \hat{P}_2) - 0}{\sqrt{\dfrac{\hat{P}(1 - \hat{P})}{n_1} + \dfrac{\hat{P}(1 - \hat{P})}{n_2} }} \sim N(0, 1) $$ where $\hat{P} = \dfrac{\text{total number of successes} }{ \text{total number of cases}}.$
#### Observed test statistic
While one could compute this observed test statistic by "hand", the focus here is on the set-up of the problem and in understanding which formula for the test statistic applies. We can use the `prop.test` function to perform this analysis for us.
```{r prop.test2}
stats::prop.test(x = table(offshore$college_grad, offshore$response),
n = nrow(offshore),
alternative = "two.sided",
correct = FALSE)
```
`prop.test` does a $\chi^2$ test here but this matches up exactly with what we would expect from the test statistic above since $Z^2 = \chi^2$ so $\sqrt{9.99} = 3.16 = z_{obs}$: The $p$-values are the same because we are focusing on a two-tailed test. The observed difference in sample proportions is 3.16 standard deviations larger than 0.
```{r infer1, include=FALSE, eval=FALSE}
inference(x = offshore$college_grad,
y = offshore$response,
est = "proportion",
success = "no opinion",
alternative = "twosided",
type = "ht",
null = 0,
method = "theoretical",
eda_plot = FALSE,
inf_plot = FALSE)
```
The $p$-value---the probability of observing a $Z$ value of 3.16 or more extreme in our null distribution---is 0.0016. This can also be calculated in R directly:
```{r pval2prop}
2 * pnorm(3.16, lower.tail = FALSE)
```
The 95% confidence interval is also stated above in the `prop.test` results.
### State conclusion
We, therefore, have sufficient evidence to reject the null hypothesis. Our initial guess that a statistically significant difference did not exist in the proportions of no opinion on offshore drilling between college educated and non-college educated Californians was not validated. We do have evidence to suggest that there is a dependency between college graduation and position on offshore drilling for Californians.
---
### Comparing results
Observing the bootstrap distribution and the null distribution that were created, it makes quite a bit of sense that the results are so similar for traditional and non-traditional methods in terms of the $p$-value and the confidence interval since these distributions look very similar to normal distributions. The conditions were not met since the number of pairs was small, but the sample data was not highly skewed. Using any of the methods whether they are traditional (formula-based) or non-traditional (computational-based) lead to similar results.
***
***
## Two means (independent samples)
### Problem statement
Average income varies from one region of the country to
another, and it often reflects both lifestyles and regional living expenses. Suppose a new graduate
is considering a job in two locations, Cleveland, OH and Sacramento, CA, and he wants to see
whether the average income in one of these cities is higher than the other. He would like to conduct
a hypothesis test based on two randomly selected samples from the 2000 Census. [Tweaked a bit from @isrs2014 [Chapter 5]]
### Competing hypotheses
#### In words
- Null hypothesis: There is no association between income and location (Cleveland, OH and Sacramento, CA).
- Alternative hypothesis: There is an association between income and location (Cleveland, OH and Sacramento, CA).
#### Another way in words
- Null hypothesis: The mean income is the **same** for both cities.
- Alternative hypothesis: The mean income is different for the two cities.
#### In symbols (with annotations)
- $H_0: \mu_{sac} = \mu_{cle}$ or $H_0: \mu_{sac} - \mu_{cle} = 0$, where $\mu$ represents the average income.
- $H_A: \mu_{sac} - \mu_{cle} \ne 0$
#### Set $\alpha$
It's important to set the significance level before starting the testing using the data. Let's set the significance level at 5% here.
### Exploring the sample data
```{r read_data4, echo=FALSE}
#download.file("http://ismayc.github.io/teaching/sample_problems/cleSac.txt",
# destfile = "data/cleSac.txt",
# method = "curl")
cleSac <- read.delim("data/cleSac.txt") %>%
rename(metro_area = Metropolitan_area_Detailed,
income = Total_personal_income) %>%
na.omit()
```
```{r summarize3}
inc_summ <- cleSac %>% group_by(metro_area) %>%
summarize(sample_size = n(),
mean = mean(income),
sd = sd(income),
minimum = min(income),
lower_quartile = quantile(income, 0.25),
median = median(income),
upper_quartile = quantile(income, 0.75),
max = max(income))
kable(inc_summ)
```
The boxplot below also shows the mean for each group highlighted by the red dots.
```{r boxplot}
ggplot(cleSac, aes(x = metro_area, y = income)) +
geom_boxplot() +
stat_summary(fun.y = "mean", geom = "point", color = "red")
```
#### Guess about statistical significance
We are looking to see if a difference exists in the mean income of the two levels of the explanatory variable. Based solely on the boxplot, we have reason to believe that no difference exists. The distributions of income seem similar and the means fall in roughly the same place.
***
### Non-traditional methods
#### Collecting summary info
Next we will assign some key values to variable names in R:
```{r stats2}
xbar_cle <- inc_summ$mean[1]
xbar_sac <- inc_summ$mean[2]
obs_diff <- xbar_sac - xbar_cle
n_cle <- inc_summ$sample_size[1]
n_sac <- inc_summ$sample_size[2]
```
#### Randomization for hypothesis test
In order to look to see if the observed sample mean for Sacramento of `r xbar_cle` is statistically different than that for Cleveland of `r xbar_sac`, we need to account for the sample sizes. Note that this is the same as looking to see if $\bar{x}_{sac} - \bar{x}_{cle}$ is statistically different than 0. We also need to determine a process that replicates how the original group sizes of `r n_cle` and `r n_sac` were selected.
We can use the idea of *randomization testing* (also known as *permutation testing*) to simulate the population from which the sample came (with two groups of different sizes) and then generate samples using *shuffling* from that simulated population to account for sampling variability.
```{r, echo=FALSE}
if(!file.exists("rds/null_distn_two_means.rds")){
set.seed(2017)
many_shuffles <- do(10000) *
(cleSac %>%
mutate(metro_area = shuffle(metro_area)) %>%
group_by(metro_area) %>%
summarize(mean_inc = mean(income))
)
null_distn_two_means <- many_shuffles %>%
group_by(.index) %>%
summarize(diffmean = diff(mean_inc))
saveRDS(object = null_distn_two_means, "rds/null_distn_two_means.rds")
} else {
null_distn_two_means <- readRDS("rds/null_distn_two_means.rds")
}
```
```{r sim4, eval=FALSE}
set.seed(2017)
many_shuffles <- do(10000) *
(cleSac %>%
mutate(metro_area = shuffle(metro_area)) %>%
group_by(metro_area) %>%
summarize(mean_inc = mean(income))
)
null_distn_two_means <- many_shuffles %>%
group_by(.index) %>%
summarize(diffmean = diff(mean_inc))
```
```{r}
ggplot(null_distn_two_means, aes(x = diffmean)) +
geom_histogram(bins = 30, color = "white")
```
We can next use this distribution to observe our $p$-value. Recall this is a two-tailed test so we will be looking for values that are greater than or equal to `r obs_diff` or less than or equal to `r -obs_diff` for our $p$-value.
```{r}
ggplot(null_distn_two_means, aes(x = diffmean)) +
geom_histogram(bins = 30, color = "white") +
geom_vline(color = "red", xintercept = obs_diff) +
geom_vline(color = "red", xintercept = -obs_diff)
```
##### Calculate $p$-value
```{r}
pvalue <- null_distn_two_means %>%
filter( (diffmean >= obs_diff) | (diffmean <= -obs_diff) ) %>%
nrow() / nrow(null_distn_two_means)
pvalue
```
So our $p$-value is `r pvalue` and we fail to reject the null hypothesis at the 5% level. You can also see this from the histogram above that we are not very far into the tail of the null distribution.
#### Bootstrapping for confidence interval
We can also create a confidence interval for the unknown population parameter $\mu_{sac} - \mu_{cle}$ using our sample data with *bootstrapping*. Here we will bootstrap each of the groups with replacement instead of shuffling. This is done using the `groups`
argument in the `resample` function to fix the size of each group to
be the same as the original group sizes of `r n_sac` for Sacramento and `r n_cle` for Cleveland.
```{r, echo=FALSE}
if(!file.exists("rds/boot_distn_two_means.rds")){
boot_means <- do(10000) *
cleSac %>%
resample(replace = TRUE, groups = metro_area) %>%
group_by(metro_area) %>%
summarize(mean_inc = mean(income))
# Next, we calculate the difference in sample means for each of the 10,000 replications:
boot_distn_two_means <- boot_means %>%
group_by(.index) %>%
summarize(diffmean = diff(mean_inc))
saveRDS(object = boot_distn_two_means, "rds/boot_distn_two_means.rds")
} else {
boot_distn_two_means <- readRDS("rds/boot_distn_two_means.rds")
}
```
```{r boot4, eval=FALSE}
boot_means <- do(10000) *
cleSac %>%
resample(replace = TRUE, groups = metro_area) %>%
group_by(metro_area) %>%
summarize(mean_inc = mean(income))
# Next, we calculate the difference in sample means for each of the 10,000 replications:
boot_distn_two_means <- boot_means %>%
group_by(.index) %>%
summarize(diffmean = diff(mean_inc))
```
```{r}
ggplot(boot_distn_two_means, aes(x = diffmean)) +
geom_histogram(bins = 30, color = "white")
```