Here we will walk through the most common statistical tests and show how they can be represented in the linear regression format. This section is based on this post. Check it out for a more in-depth description of the non-parametric alternatives.


One sample t-test

The test evaluates if the mean of a continuous variable is different from 0. It is equivalent to a linear regression with no predictor, that tests if the intercept is different from 0:

\(y = \beta_0 \qquad \mathcal{H}_0: \beta_0 = 0\)

 

# number of observations
n <- 50

# set seed
set.seed(123)

# create variable with mean 1
y <- rnorm(n = n, mean = 1)

# run t test
(t <- t.test(y))
## 
##  One Sample t-test
## 
## data:  y
## t = 7.9, df = 49, p-value = 2.7e-10
## alternative hypothesis: true mean is not equal to 0
## 95 percent confidence interval:
##  0.77127 1.29753
## sample estimates:
## mean of x 
##    1.0344
# run equivalent linear regression
(lm_t <- summary(lm(y ~ 1)))
## 
## Call:
## lm(formula = y ~ 1)
## 
## Residuals:
##    Min     1Q Median     3Q    Max 
## -2.001 -0.594 -0.107  0.664  2.135 
## 
## Coefficients:
##             Estimate Std. Error t value Pr(>|t|)    
## (Intercept)    1.034      0.131     7.9  2.7e-10 ***
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
## 
## Residual standard error: 0.926 on 49 degrees of freedom

 

Let’s put the results from both tests together to compare them more closely:

model estimate p.value t
t-test 1.0344 0 7.9
lm 1.0344 0 7.9

 

Note that, as there are no predictor in the model, we used a ‘1’ in the place for predictors in the model formula (‘y ~ 1’).


Paired t-test

A paired t-test evaluates if the mean difference between two numeric variables is 0:

\(y1 - y2 = \beta_0 \qquad \mathcal{H}_0: \beta_0 = 0\)

 

The correspondent linear model is the same than that for the one-sample t-test, but the input variable is the difference between the two variables (y1 - y2):

set.seed(123)

# sample size
n <- 50

# variable with mean 1
y1 <- rnorm(n = n, mean = 1)

# variable with mean 3
y2 <- rnorm(n = n, mean = 1.4)

# run paired t test
paired_t <- t.test(y1, y2, paired = TRUE)

paired_t
## 
##  Paired t-test
## 
## data:  y1 and y2
## t = -2.75, df = 49, p-value = 0.0084
## alternative hypothesis: true mean difference is not equal to 0
## 95 percent confidence interval:
##  -0.88659 -0.13742
## sample estimates:
## mean difference 
##          -0.512
# difference between the 2 variables
diff_y <- y1 - y2

# run model
lm_paired_t <- summary(lm(formula = diff_y ~ 1))

lm_paired_t
## 
## Call:
## lm(formula = diff_y ~ 1)
## 
## Residuals:
##    Min     1Q Median     3Q    Max 
## -2.600 -0.701  0.112  1.035  2.909 
## 
## Coefficients:
##             Estimate Std. Error t value Pr(>|t|)   
## (Intercept)   -0.512      0.186   -2.75   0.0084 **
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
## 
## Residual standard error: 1.32 on 49 degrees of freedom
model estimate p.value t
paired t-test -0.512 0.0084 -2.7468
lm -0.512 0.0084 -2.7468

 


Two means t-test

Also called independent t-test. Evaluates if the means of the two variables are different. The null hypothesis is that the difference is 0:

\(y = \beta_0 + \beta_1 * x1 \qquad \mathcal{H}_0: \beta_1 = 0\)

# set seed
set.seed(123)

# number of observations
n <- 50
b0 <- -4
b1 <- 3
error <- rnorm(n = n, mean = 0, sd = 1)

# random variables
x1_num <- rbinom(n = n, size = 1, prob = 0.5)
y <- b0 + b1 * x1_num + error

x1 <- factor(x1_num, labels = c("a", "b"))

# create data frame
xy_data_cat <- data.frame(x1, x1_num, y)

# run paired t test
indep_t <- t.test(xy_data_cat$y[xy_data_cat$x1 == "a"], xy_data_cat$y[xy_data_cat$x1 ==
    "b"])

indep_t
## 
##  Welch Two Sample t-test
## 
## data:  xy_data_cat$y[xy_data_cat$x1 == "a"] and xy_data_cat$y[xy_data_cat$x1 == "b"]
## t = -12.3, df = 47.9, p-value <2e-16
## alternative hypothesis: true difference in means is not equal to 0
## 95 percent confidence interval:
##  -3.7518 -2.6957
## sample estimates:
## mean of x mean of y 
##  -4.07748  -0.85371
# run regression model
lm_indep_t <- summary(lm(formula = y ~ x1, data = xy_data_cat))

lm_indep_t
## 
## Call:
## lm(formula = y ~ x1, data = xy_data_cat)
## 
## Residuals:
##    Min     1Q Median     3Q    Max 
## -2.113 -0.594 -0.136  0.552  2.246 
## 
## Coefficients:
##             Estimate Std. Error t value Pr(>|t|)    
## (Intercept)   -4.077      0.186   -22.0   <2e-16 ***
## x1b            3.224      0.263    12.3   <2e-16 ***
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
## 
## Residual standard error: 0.928 on 48 degrees of freedom
## Multiple R-squared:  0.758,  Adjusted R-squared:  0.753 
## F-statistic:  151 on 1 and 48 DF,  p-value: <2e-16
model estimate p.value t
2 means t-test 3.2238 0 -12.276
lm 3.2238 0 12.276

 


Three or more means: one-way ANOVA

Very similar to the two means t-test but with three or more levels in the categorical variable:

\(\hat{Y} \sim \beta_{o} + \beta_{1} * x_{1} + \beta_{2} * x_{2} + ... \qquad \mathcal{H}_0: y = \beta_0\)

 

So it is a multiple regression model. The categorical variable is dummy coded so it gets split into indicator predictors (\(x_i\) is either \(x=0\) or \(x=1\)).

A data set with a 3-level categorical variable can be simulated like this:

# set seed
set.seed(123)

# number of observations
n <- 50
b0 <- -4
b1 <- 3
error <- rnorm(n = n, mean = 0, sd = 1)

# random variables
x1_num <- rbinom(n = n, size = 2, prob = c(0.33, 0.33))
y <- b0 + b1 * x1_num + error

x1 <- factor(x1_num, labels = c("a", "b", "c"))

# create data frame
xy_data_cat2 <- data.frame(x1, y)

head(xy_data_cat2)
x1 y
b -1.5605
a -4.2302
b 0.5587
c 2.0705
b -0.8707
b 0.7151
# ANOVA function
anv_1w <- anova(aov(formula = y1 ~ x1, data = xy_data_cat2))

anv_1w
Df Sum Sq Mean Sq F value Pr(>F)
2 4.3728 2.1864 2.7307 0.0755
47 37.6318 0.8007 NA NA
# linear regression
lm_anv_1w <- summary(lm(formula = y1 ~ x1, data = xy_data_cat2))

lm_anv_1w
## 
## Call:
## lm(formula = y1 ~ x1, data = xy_data_cat2)
## 
## Residuals:
##     Min      1Q  Median      3Q     Max 
## -1.6721 -0.5195 -0.0289  0.4839  2.3266 
## 
## Coefficients:
##             Estimate Std. Error t value Pr(>|t|)    
## (Intercept)    0.842      0.187    4.51  4.2e-05 ***
## x1b            0.565      0.277    2.04    0.047 *  
## x1c           -0.141      0.367   -0.38    0.703    
## ---
## Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
## 
## Residual standard error: 0.895 on 47 degrees of freedom
## Multiple R-squared:  0.104,  Adjusted R-squared:  0.066 
## F-statistic: 2.73 on 2 and 47 DF,  p-value: 0.0755
model F.statistic p.value
1 way anova 2.7307 0.0755
lm 2.7307 0.0755

 


Chi-square for contingency tables

This test looks to find and association between two categorical variables based on the co-occurrence of the categories, measured as counts. This is what a contingency table is, counts for all combinations of categories between two variables. For instance deaths by age of passenger of the Titanic:

# Contingency table modified from
# https://lindeloev.github.io/tests-as-linear/
mood_data <- matrix(data = c(100, 70, 30, 32, 110, 120), nrow = 2,
    dimnames = list(handedness = c("left_handed", "right_handed"),
        mood = c("happy", "meh", "sad")))

mood_data
##               mood
## handedness     happy meh sad
##   left_handed    100  30 110
##   right_handed    70  32 120

 

Although the coding is a bit more elaborated it can be narrow down to log-linear two-way ANOVA model:

\(log(y_i) = log(N) + log(\alpha_i) + log(\beta_j) + log(\alpha_i\beta_j)\)

 

The regression coefficients are \(A_i\) and \(B_i\). \(\alpha_i\) and \(\beta_j\) are the proportions of counts for the two levels of the binary categorical variable (one for each of the levels in the second categorical variable). Fortunately, there is an implementation of this model from the R package ‘MASS’ that simplifies coding:

# Built-in chi-square. It requires matrix format.
chi_mod <- chisq.test(mood_data)

chi_mod
## 
##  Pearson's Chi-squared test
## 
## data:  mood_data
## X-squared = 5.1, df = 2, p-value = 0.078
# convert to long
mood_data_long <- as.data.frame(as.table(mood_data))

# log linear model from MASS package
log_lm_mod <- MASS::loglm(Freq ~ handedness + mood, data = mood_data_long)

log_lm_mod
## Call:
## MASS::loglm(formula = Freq ~ handedness + mood, data = mood_data_long)
## 
## Statistics:
##                     X^2 df P(> X^2)
## Likelihood Ratio 5.1199  2 0.077308
## Pearson          5.0999  2 0.078087
model X.squared p.value
chi-square 5.0999 0.0781
log lm 5.0999 0.0781

 

‘Non-parametric’ alternatives as linear regression models

We will only feature one example of a ‘non-parametric’ test. Nonetheless, this single example should be enough to demonstrate the logic behind ‘non-parametric’ procedures. Most ‘non-parametric’ rely on a simple data transformation trick: rank transforming responses. Rank transformations converts values to an index according to their position when sorted by magnitude:

# simulate variable
set.seed(123)
y <- round(rnorm(n = 5), 2)

y
## [1] -0.56 -0.23  1.56  0.07  0.13
# convert to ranks
rank_y <- rank(y)

rank_y
## [1] 1 2 5 3 4

 

Wilcoxon test for two means

A.k.a Mann-Whitney test, is the non-parametric alternative to a two-mean t test:

# set seed
set.seed(123)

# number of observations
n <- 50
y <- rnorm(n = n, mean = 0.2, sd = 1)

# create data frame
y_data <- data.frame(y)


y
##  [1] -0.3604756 -0.0301775  1.7587083  0.2705084  0.3292877
##  [6]  1.9150650  0.6609162 -1.0650612 -0.4868529 -0.2456620
## [11]  1.4240818  0.5598138  0.6007715  0.3106827 -0.3558411
## [16]  1.9869131  0.6978505 -1.7666172  0.9013559 -0.2727914
## [21] -0.8678237 -0.0179749 -0.8260044 -0.5288912 -0.4250393
## [26] -1.4866933  1.0377870  0.3533731 -0.9381369  1.4538149
## [31]  0.6264642 -0.0950715  1.0951257  1.0781335  1.0215811
## [36]  0.8886403  0.7539177  0.1380883 -0.1059627 -0.1804710
## [41] -0.4947070 -0.0079173 -1.0653964  2.3689560  1.4079620
## [46] -0.9231086 -0.2028848 -0.2666554  0.9799651  0.1166309
# Wilcoxon / Mann-Whitney U
wilcx_mod <- wilcox.test(y_data$y)

wilcx_mod
## 
##  Wilcoxon signed rank test with continuity correction
## 
## data:  y_data$y
## V = 797, p-value = 0.12
## alternative hypothesis: true location is not equal to 0
signed_rank = function(x) sign(x) * rank(abs(x))

# As linear model with our dummy-coded group_y2:
wicx_lm_mod <- lm(signed_rank(y) ~ 1, data = y_data)  # compare to 

summary(wicx_lm_mod)
## 
## Call:
## lm(formula = signed_rank(y) ~ 1, data = y_data)
## 
## Residuals:
##    Min     1Q Median     3Q    Max 
## -53.38 -24.13   0.12  25.37  43.62 
## 
## Coefficients:
##             Estimate Std. Error t value Pr(>|t|)
## (Intercept)     6.38       4.09    1.56     0.12
## 
## Residual standard error: 28.9 on 49 degrees of freedom
model p.value
1 mean wilcoxon 0.1248
wilcoxon_lm 0.1248

 


References

 


 

Session information

## R version 4.2.2 Patched (2022-11-10 r83330)
## Platform: x86_64-pc-linux-gnu (64-bit)
## Running under: Ubuntu 20.04.5 LTS
## 
## Matrix products: default
## BLAS:   /usr/lib/x86_64-linux-gnu/blas/libblas.so.3.9.0
## LAPACK: /usr/lib/x86_64-linux-gnu/lapack/liblapack.so.3.9.0
## 
## locale:
##  [1] LC_CTYPE=es_ES.UTF-8       LC_NUMERIC=C              
##  [3] LC_TIME=es_CR.UTF-8        LC_COLLATE=es_ES.UTF-8    
##  [5] LC_MONETARY=es_CR.UTF-8    LC_MESSAGES=es_ES.UTF-8   
##  [7] LC_PAPER=es_CR.UTF-8       LC_NAME=C                 
##  [9] LC_ADDRESS=C               LC_TELEPHONE=C            
## [11] LC_MEASUREMENT=es_CR.UTF-8 LC_IDENTIFICATION=C       
## 
## attached base packages:
## [1] stats     graphics  grDevices utils     datasets  methods  
## [7] base     
## 
## other attached packages:
## [1] viridis_0.6.3      viridisLite_0.4.2  dplyr_1.1.0       
## [4] tidyr_1.3.0        readxl_1.4.1       kableExtra_1.3.4  
## [7] knitr_1.42         ggplot2_3.4.2      RColorBrewer_1.1-3
## 
## loaded via a namespace (and not attached):
##   [1] minqa_1.2.5         colorspace_2.1-0    ellipsis_0.3.2     
##   [4] rsconnect_0.8.29    sjlabelled_1.2.0    estimability_1.4.1 
##   [7] fs_1.6.2            rstudioapi_0.14     farver_2.1.1       
##  [10] remotes_2.4.2       fansi_1.0.4         mvtnorm_1.1-3      
##  [13] xml2_1.3.4          splines_4.2.2       cachem_1.0.8       
##  [16] sjmisc_2.8.9        pkgload_1.3.2       jsonlite_1.8.4     
##  [19] nloptr_2.0.3        ggeffects_1.2.2     broom_1.0.4        
##  [22] shiny_1.7.3         compiler_4.2.2      httr_1.4.6         
##  [25] sjstats_0.18.2      emmeans_1.8.6       backports_1.4.1    
##  [28] assertthat_0.2.1    Matrix_1.5-1        fastmap_1.1.1      
##  [31] cli_3.6.1           later_1.3.1         formatR_1.12       
##  [34] htmltools_0.5.5     prettyunits_1.1.1   tools_4.2.2        
##  [37] lmerTest_3.1-3      coda_0.19-4         gtable_0.3.3       
##  [40] glue_1.6.2          Rcpp_1.0.10         carData_3.0-5      
##  [43] cellranger_1.1.0    jquerylib_0.1.4     vctrs_0.6.2        
##  [46] sjPlot_2.8.14       svglite_2.1.0       nlme_3.1-162       
##  [49] insight_0.19.2      xfun_0.39           stringr_1.5.0      
##  [52] ps_1.7.5            lme4_1.1-33         rvest_1.0.3        
##  [55] mime_0.12           miniUI_0.1.1.1      lifecycle_1.0.3    
##  [58] devtools_2.4.5      klippy_0.0.0.9500   MASS_7.3-58.2      
##  [61] scales_1.2.1        ragg_1.2.4          promises_1.2.0.1   
##  [64] yaml_2.3.7          memoise_2.0.1       gridExtra_2.3      
##  [67] sass_0.4.6          stringi_1.7.12      highr_0.10         
##  [70] bayestestR_0.13.1   boot_1.3-28         pkgbuild_1.3.1     
##  [73] rlang_1.1.1         pkgconfig_2.0.3     systemfonts_1.0.4  
##  [76] evaluate_0.21       lattice_0.20-45     purrr_1.0.1        
##  [79] htmlwidgets_1.5.4   labeling_0.4.2      tidyselect_1.2.0   
##  [82] processx_3.8.1      magrittr_2.0.3      R6_2.5.1           
##  [85] generics_0.1.3      profvis_0.3.7       pillar_1.9.0       
##  [88] withr_2.5.0         mgcv_1.8-41         abind_1.4-5        
##  [91] tibble_3.2.1        performance_0.10.3  modelr_0.1.11      
##  [94] crayon_1.5.2        car_3.1-2           utf8_1.2.3         
##  [97] rmarkdown_2.21      urlchecker_1.0.1    usethis_2.1.6      
## [100] grid_4.2.2          isoband_0.2.7       callr_3.7.3        
## [103] digest_0.6.31       webshot_0.5.4       xtable_1.8-4       
## [106] httpuv_1.6.6        numDeriv_2016.8-1.1 textshaping_0.3.6  
## [109] munsell_0.5.0       bslib_0.4.2         sessioninfo_1.2.2