- Sorenson Impact Center

# Bootstrap Statistics with Tidymodels to Compare Bicycle Helmets

Bootstrapping has long been one of my favorite statistical procedures. The nonparametric version requires few assumptions, and it shares attributes with both simulation and common Bayesian models, both of which I love. At the end of the day I wonder, why settle for a point estimate and two values for the confidence intervals when you can create a distribution and *visualize* your CIs? When my friend Rich first introduced me to bootstrapping, he said that if statistics had been invented in the computer age, this is where most classes would start. I agree.

Tidymodels has provided an excellent framework for applying bootstrapping using tidyverse principles. The rsample package will be the focus of this post.

I scraped data from Virginia Tech’s helmet ratings. I’m sincerely grateful that they not only release this data, but have also answered my questions about methodology. If all test labs adopted VT’s open-source ethos, consumers would be much more confident about the safety of their helmets.

The motivating example is to estimate the affect of a new helmet technology called WaveCel, which is claimed to significantly reduce the probability of a concussion by absorbing angled impacts better. As it happens, I have already crashed twice in a WaveCel helmet while riding my mountain bike (I know, I know. I’ve dialed it back since then). In both cases, I avoided what may have been a more serious brain injury. But what do the test data say?

Virginia Tech assigns scores to helmets based on a testing methodology that accounts for the frequency and severity of impacts. A lower score in their system represents a reduced probability of common injuries. Importantly for our analysis, the scores are not normally distributed.

I am most interested in the difference between two technologies, WaveCel and MIPS, both of which reduce torsional impacts. WaveCel is such a new technology that there are only four helmets that use it currently on the market–a small sample size, to be sure. Fortunately, all four have been tested by VT; and their scores are similar enough that it is possible to see the difference between WaveCel, its more established rival MIPS, and the standard helmets that mostly protect against linear impacts.

What is obvious in this plot is that MIPS and WaveCel are right on par with each other, and that both technologies test better than regular helmets. Some say that VT’s testing protocols favor MIPS and WaveCel by coupling the headform more tightly to the helmet. It would be great if other labs released their data. But for now this is the most comprehensive way to compare helmets.

I could perform a simple t-test of the means, which would show WaveCel with a slight edge over MIPS. But I wanted to go a step further and control for the helmet types, which affect test performance too.

### Bootstrap the Tidy Way

Tidymodels is a collection of packages that automate common modeling practices, simplify tuning and training, and make it easier to store outputs in a tidy format. I find that the unique nomenclature can be confusing (am I baking this thing, or juicing it?), but in general love how it streamlines everything from data preparation to cross validation.

For this problem, I begin by setting up a bootstrap of 3,000 resamples. Each split is the same length of the original, but with random samples taken with replacement, such that any given helmet could be included more than once.

set.seed(1876) helmet_bootraps <- bootstraps(helmets, times = 3e3, apparent = TRUE)

You can access any of the splits and examine how it differs from the original data.

head(as_tibble(helmet_bootraps$splits[[55]]))

## # A tibble: 6 x 6 ## V1 id score style mips wavecel ## <chr> <dbl> <dbl> <chr> <lgl> <lgl> ## 1 Lazer Blade 74 19.2 Road FALSE FALSE ## 2 Nutcase Street 92 22.7 Urban FALSE FALSE ## 3 Giro Revel 87 20.9 Multi-sport FALSE FALSE ## 4 Bell Draft MIPS 53 15.6 Road TRUE FALSE ## 5 Schwinn Chic 94 22.9 Urban FALSE FALSE ## 6 Bontrager Specter WaveCel 8 10.8 Road FALSE TRUE

### Map Functions

Now for the cool part: because the splits are stored as lists within a tibble, you can easily map any type of function to the data. For this example, I’ve written three functions:

The mean of MIPS scores

A linear model with type and technology as independent variables

The difference between the coefficients for MIPS and WaveCel in that model

#1 calc_mips_mean <- function(split){ dat <- analysis(split) %>% filter(mips) %>% pull(score) # Put it in this tidy format to use int_pctl return(tibble( term = "mean", estimate = mean(dat), std.err = sd(dat)/sqrt(length(dat)))) } # 2 base_model <- function(split){ lm(score ~ style + mips + wavecel, data = analysis(split)) %>% tidy() } # 3 beta_diff <- function(split){ model <- lm(score ~ style + mips + wavecel, data = analysis(split)) # Put it in this tidy format to use int_pctl return(tibble( term = "diff_mips_wavecel", estimate = model$coefficients["wavecelTRUE"] - model$coefficients["mipsTRUE"], std.err = NA_real_)) }

And, as I described above, now it’s just a matter of mapping these to the data, which will add three new tibbles to each of the bootstraps.

helmet_stats <- helmet_bootraps %>% mutate( mips_mean = map(splits, calc_mips_mean), coef_info = map(splits, base_model), coef_diff = map(splits, beta_diff)) head(helmet_stats)

## # A tibble: 6 x 5 ## splits id mips_mean coef_info coef_diff ## * <list> <chr> <list> <list> <list> ## 1 <split [99/30]> Bootstrap0001 <tibble [1 × 3]> <tibble [6 × 5… <tibble [1 × 3… ## 2 <split [99/31]> Bootstrap0002 <tibble [1 × 3]> <tibble [6 × 5… <tibble [1 × 3… ## 3 <split [99/37]> Bootstrap0003 <tibble [1 × 3]> <tibble [6 × 5… <tibble [1 × 3… ## 4 <split [99/37]> Bootstrap0004 <tibble [1 × 3]> <tibble [6 × 5… <tibble [1 × 3… ## 5 <split [99/39]> Bootstrap0005 <tibble [1 × 3]> <tibble [6 × 5… <tibble [1 × 3… ## 6 <split [99/35]> Bootstrap0006 <tibble [1 × 3]> <tibble [6 × 5… <tibble [1 × 3…

With this meta-tibble, I can make us of handy functions from the rsample package to calculate things like percentile intervals, T-intervals, and BCa, which are useful for obtaining confidence intervals of bootstrapped statistics.

int_pctl(helmet_stats, mips_mean)

## # A tibble: 1 x 6 ## term .lower .estimate .upper .alpha .method ## <chr> <dbl> <dbl> <dbl> <dbl> <chr> ## 1 mean 12.6 13.2 13.9 0.05 percentile

Finally, I unnest the data to visualize it.

helmet_coefs <- helmet_stats %>% unnest(coef_info) helmet_coefs %>% select(estimate, term) %>% filter(term %in% c("wavecelTRUE", "mipsTRUE")) %>% ggplot(aes(estimate, fill = term)) + geom_histogram(alpha = 0.7, position="identity") + scale_fill_manual(values=c("#999999", "#E69F00")) + labs(x = "Estimated Impact of the Technology", y = "Count /3k Bootstrapped Models", title = "WaveCel is better, but there is overlap")

This plot is why I appreciate bootstrapping. It’s a visual representation of thousands of model coefficients, each from a different sample of the original data, showing the range of outcomes for both technologies.

To get an intuition of what’s happening, imagine taking data from one of the more extreme results, say the far left tail on WaveCel.

extreme_result <- helmet_coefs %>% filter(term == "wavecelTRUE") %>% arrange(estimate) %>% slice(1) as_tibble(extreme_result$splits[[1]]) %>% filter(wavecel) %>% head()

## # A tibble: 3 x 6 ## V1 id score style mips wavecel ## <chr> <dbl> <dbl> <chr> <lgl> <lgl> ## 1 Bontrager Specter WaveCel 8 10.8 Road FALSE TRUE ## 2 Bontrager Specter WaveCel 8 10.8 Road FALSE TRUE ## 3 Bontrager Specter WaveCel 8 10.8 Road FALSE TRUE

In this unusual case, the data randomly contained three Specter models, which is the best performing WaveCel helmet. In most cases, however, the data will contain a balanced mix of representative models and types. WaveCel is more dispersed because of the small original sample, which underlies subsequent resamples.

### Bootstrapping the Differences

To get a final estimate of the *difference* between MIPS and WaveCel, I will use the percentile interval on the delta between the coefficients. That’s a lot of nouns. What I mean is, for each bootstrap sample I run a regression and subtract the WaveCel effect from the MIPS effect (controlling for helmet type). The percentile interval is a way to show an upper and lower bound that contain about 95% of those values.

int_pctl(helmet_stats, coef_diff)

## # A tibble: 1 x 6 ## term .lower .estimate .upper .alpha .method ## <chr> <dbl> <dbl> <dbl> <dbl> <chr> ## 1 diff_mips_wavecel -2.55 -1.18 0.274 0.05 percentile

Based on this, WaveCel is probably better at VT’s tests. This shows an upper value that is above zero, however, so we cannot rule out the possibility that MIPS is better. Once again, having a tidy tibble is nice, because I can filter the data to values in the 95% interval and see what portion of those are below zero (i.e., favor WaveCel).

proportion <- helmet_stats %>% unnest(coef_diff) %>% filter(estimate >= -2.551318 & estimate <= 0.2744458) %>% group_by(id) %>% mutate(pos = sum(estimate > 0), neg = sum(estimate < 0)) sum(proportion$neg) / nrow(proportion)

## [1] 0.9782918

This shows that WaveCel held an edge in 98% of the models under consideration.

### Conclusion

Bootstrapping suggests that WaveCel is equal to a 1.17 point advantage on Virginia Tech’s tests relative to MIPS. Based on my read of VT’s methodology, this translates to about a 2.5% decrease in the probability of a concussion (they appear to have a scale of 0 - 47.4 representing common impacts). This does not seem like much. Indeed, separate tests led by WaveCel’s inventors showed a much larger effect. When it comes to preventing brain injury, however, even small gains are within my region of practical significance.

In any case, bootstrapping is a useful tool for visualizing and estimating how confident to be in model results. Tidymodels, and specifically the rsample package, provide an easy platform for working with bootstrapped samples.

*This post by *__Daniel Hadley__*, Managing Director of Data Science, originally appeared on *__danielphadley.com__*.*