MLB Injuries - Introductory Time Series Analysis

Time series plots
Time series decomposition
Residual analysis
Simple forecasting
Exploring MLB injury data through time series analysis and forecasting.
Author
Affiliation

Jonathan Lieb

Baylor University

Published

November 26, 2024

  • This module would be suitable for an in-class lab or take-home assignment in an intermediate statistics course.

  • It assumes a familiarity with the RStudio Environment and R programming language.

  • Students should be provided with the following data file (.csv) and Quarto document (.qmd) to produce visualizations and write up their answers to each exercise. Their final deliverable is to turn in an .html document produced by “Rendering” the .qmd.

  • Posit Cloud (via an Instructor account) or Github classroom are good options for disseminating files to students, but simply uploading files to your university’s course management system works, too.

  • The data for the mlb_injuries_monthly.csv file was derived from data found on prosportstransactions.com. The original data from the site contained rows of observations showing each transaction that involved sending a player to the injured list or bringing a player off of the injured list. The data was then filtered to include only times when a player was sent to the injured list and aggregated by month. The 2000 season was originally included in the data, but was removed after incomplete data was found for that season.

  • The data for the tj_surgeries_mlb_milb.csv file was derived from data accumulated by @MLBPlayerAnalys over many years. The original data lists any reported Tommy John surgery for a player now playing in the MLB or MiLB. This data was aggregated by year to produce the data used in this module.

Welcome video

Introduction

In this module, you will explore Major League Baseball (MLB) injury data through time series analysis and forecasting. The dataset contains monthly injury counts from 2001 to 2023. You will learn how to create time series plots, decompose time series data, and forecast future injury counts.

Coming off of COVID-19 breaks, the MLB saw a rash of injuries in the shortened 2020 season and the 2021 season.

In the image below Washington Nationals star shortstop Trea Turner attempts to shake off a hit-by-pitch injury in the 9th inning of a game against the Atlanta Braves April 26th, 2021.

Image source: All-Pro Reels from District of Columbia, USA, CC BY-SA 2.0, via Wikimedia Commons

Injuries significantly impact a team’s performance and can be costly to the organization. In fact, at the end of the 2023 season MLB teams combined for over $1 billion worth of salaries on the injured list. The Angels single-handedly had over $145 million worth of injured player salaries, more than than three times as much money as their 26-man roster was making. This contributed heavily to their 73-89 record. On the flip side, 7 of the 11 teams with the fewest amount of money on the injured list made the playoffs. Health matters to teams, players, coaches, and the league. When injuries accumulate for teams it forces teams to play worse players-harming performance- or acquire new players at a cost-harming the team’s bottom line.

If you’d like to see more information about MLB salary breakdowns, check out Spotrac’s webpage.

Despite the rise of analytics and specialized training for players, MLB injuries feel as though they are happening more and more frequently. If this is the case there are could be major financial repercussions in the future as well as a loss in the quality of baseball being played. In this module we will investigate the issue of injuries in the MLB and see if injuries are truly on the rise in the MLB? Additionally we will look at when injuries are occurring to determine when teams should be most cautious. Are there certain months where injuries are more prevalent and what could cause injuries to occur then? Lastly, we will look at whether the amount of future injuries could be predicted using time series data. Let’s dig into the data and find out what we can learn!

Learning Objectives

By the end of this module you should be able to:

  • Create time series plots in R

  • Identify trends, seasonality, cycles, and outliers in time series data

  • Check for autocorrelation in time series data

  • Decompose time series data into trend, seasonal, and residual components

  • Interpret residual plots to assess model fit

  • Split time series data into training and test sets to evaluate forecasting models

  • Use basic forecasting techniques to predict future injury counts

  • Assess the accuracy of forecasts





NOTE: Time series analysis is a powerful tool for understanding trends and patterns in data. It is used in many fields, including finance, economics, and epidemiology. In this module, we will focus on time series analysis in the context of MLB injuries.

During this lab we will explore the following research questions:

  • What are the trends in MLB injuries over time?
  • Are there seasonal patterns in MLB injuries?
  • Can we forecast future injury counts?

Getting started: MLB Injury data

We will begin our time series analysis by loading the MLB injury data. The dataset contains monthly injury counts from 2001 to 2023.

Run the following code chunk to load the necessary libraries and the MLB injury data. In this code chunk the data is also converted to a tsibble object, which is a time series object that includes a time index and one or more columns of data. This makes it easier to work with time series data in R.

library(fpp3)
library(tidyverse)
library(patchwork)

injuries <- read_csv("mlb_injuries_monthly.csv") |> 
  mutate(Month = yearmonth(Month)) |> 
  as_tsibble(index = Month)

TIP: As you follow along in the lab, you should run each corresponding code chunk in your .qmd document.

TIP: You will most likely need to install the fpp3 package to complete this lab. You can install it by running the following code chunk:

install.packages("fpp3")

By using the glimpse function as shown below, we can see structure of the injuries data. The Month column contains the month and year of each observation, such as “2001 Jan”. This column is formatted as a date already to make time series analysis easier. The Count column contains the number of players sent to the injured list each month. This is the cumulative total of players sent to the IL across the entire MLB in that month. The value of Count is what we are concerned with and will be forecasting in this module.

glimpse(injuries)
Rows: 276
Columns: 2
$ Month <mth> 2001 Jan, 2001 Feb, 2001 Mar, 2001 Apr, 2001 May, 2001 Jun, 2001…
$ Count <dbl> 0, 0, 57, 59, 79, 92, 62, 57, 31, 2, 0, 0, 0, 0, 81, 68, 87, 77,…
Variable Descriptions
Variable Description
Month The month and year
Count The number of players sent to the IL that month
Terms to Know

Before proceeding with the time series analysis, let’s make sure we understand some baseball tidbits that will help us save our analysis.

What is the Injured List (IL)? The injured list (IL) is a list of players who are unable to play due to injury. When a player is placed on the IL, they are not eligible to play in games for a specified period, usually 10, 15, or 60 days. The IL is used to manage player injuries and ensure that teams have enough healthy players to compete. Players on the IL do not count against the team’s active roster limit.

For this lab, we will only be analyzing the number of players sent to the IL each month to understand trends and patterns in MLB injuries over time. We will not be looking at types of injuries, the severity of injuries, or the impact of injuries on team performance.

When is the MLB season? The MLB season typically runs from late March to September, with the playoffs taking place in October. The regular season consists of 162 games for each team, with games played almost every day. The season is divided into two halves, with the All-Star Game (teams get almost a full week off for the All-Star break) taking place in July. Spring training takes place in February and March.

Click here to learn more about the 10-day Injured List in Major League Baseball. Other types of ILs can also be found on the site.

Visualizing Time Series Data

The first step in any time series analysis is to visualize the data. This helps us understand the underlying patterns and trends in the data. There are four major things that we are looking for when we check our time series visualizations:

  1. Trend: A long-term increase or decrease in the data.

  2. Seasonality: Repeating patterns based on a specific time period.

  3. Cyclicity: Repeating patterns that are not based on a specific time period.

  4. Outliers: Unusual data points that do not fit the normal pattern.

We will create a time series plot of the monthly injury counts to see how the number of injuries has changed over time.

TIP: The autoplot() function is a convenient way to create time series plots in R. It automatically selects an appropriate plot type based on the data provided.

autoplot(injuries, Count) +
  labs(title = "Monthly MLB Injury Counts",
       x = "Year",
       y = "Number of Injuries")

Exercise 1: Interpreting a Time Series Plot
  1. What is the general trend in MLB injuries over time? Are injuries increasing, decreasing, or staying relatively constant?

  2. Is there a point in time where the trend appears to change? If so, when and how does it change?

  3. Does there appear to be any seasonality in the data? If so which months have the highest injury counts and which months have the lowest injury counts?

  4. Are there obvious non-seasonal cycles in the data? If so, what might explain them?

  5. Are there any obvious outliers in the data (Are there obvious breaks or deviations from the normal trends and patterns)? If so, when do they occur and what might explain them?

Autocorrelation

Autocorrelation is a measure of the linear relationship between lagged values of a time series. It can be used to help determine if there is trend or seasonality, as well as what time the seasonality is occurring at. If the autocorrelation is high for a particular lag, it suggests that the value at that lag is directly related to the value at the current time point. If a value is significantly negative, it suggests that the value at that lag is inversely related to the value at the current time point.

TIP: Lag is the number of time periods that separate the current observation from the previous observation. For example, a lag of 1 means that the current observation is compared to the previous observation, a lag of 2 means that the current observation is compared to the observation two time periods ago, and so on.

For example if there are high autocorrelations at lags 1, 2, and 3, it suggests that there is a trend in the data, since the value at time t is related to the values at times t-1, t-2, and t-3.

If the autocorrelation is high at multiples of a particular lag, such as lag 12, 24, 36, …, it could suggest that the data has a monthly seasonal pattern. If the data was based on daily observations, a high autocorrelation at lag 7 would suggest a weekly pattern.

If the autocorrelation is not significant at most of the lags, with none of the autocorrelations greatly exceeding the significance bounds, it suggests that the data is white noise. For a white noise series, we would expect the autocorrelations to be close to zero for all lags and the variation to be random.

NOTE: When performing hypothesis tests we set an alpha level, which is the probability of rejecting the null hypothesis when it is true. The default alpha level is 0.05, which means we have a 5% chance of rejecting the null hypothesis when it is true. When we do multiple tests, such as in an autocorrelation plot, the probability of making at least one Type I error (false positive) increases and can be written as \(1 - (1 - \alpha)^n\), where \(n\) is the number of tests. If you conduct 20 tests, the probability of making at least one Type I error is \(1 - (1 - 0.05)^{20} = 0.64\). This helps us understand that a significant result in one of the lags may be due to chance, and the time series may still be white noise.

We can look at an autocorrelation plot using the ACF() function.

injuries |> 
  ACF(Count) |> 
  autoplot() +
  labs(title = "Autocorrelation of Monthly MLB Injury Counts",
       x = "Lag",
       y = "Autocorrelation")

Exercise 2: Autocorrelation Plots
  1. Does the autocorrelation plot suggest any trends in the data? What point(s) in the plot suggest this?

  2. Does the autocorrelation plot suggest any seasonal patterns in the data? What point(s) in the plot suggest this?

  3. Does the autocorrelation plot suggest that the data is white noise? Why or why not?

Decomposition

Oftentimes when performing a time series analysis we want to decompose the data into its trend, seasonal, and random components to see each part individually.

Additive Decomposition is a decomposition into these three components where the observed data is the sum of the trend, seasonal, and random components.

\[y_t = T_t + S_t + R_t\]

Speaking of breaking things down, when looking at injured players by team it can be found that since 2001 the Dodgers have had the most players sent to the injured list with 708. On the opposite end of the spectrum the White Sox have had the fewest players sent to the injured list with 398.

Joe Crede of the Chicago White Sox is hit by a pitch during a game in 2006. The White Sox have had the fewest injuries since 2001.

Image Source: User SecondPrint Productions (Original version) User Mattingly23 (Crop), CC BY 2.0, via Wikimedia Commons

Where:

  • \(y_t\) is the observed data at time \(t\)
  • \(T_t\) is the trend component at time \(t\)
  • \(S_t\) is the seasonal component at time \(t\)
  • \(R_t\) is the random component at time \(t\)

This method is useful when the seasonal variation around the trend-cycle doesn’t change as the level of the series changes.

However, if the seasonal variation increases as the level of the series increases, then multiplicative decomposition is more appropriate. In this case, the observed data is the product of the trend, seasonal, and random components.

\[y_t = T_t \times S_t \times R_t\]

Classical Decomposition

There are two common methods for decomposing time series data: classical decomposition and STL decomposition. In this module we will focus on classical decomposition.

Classical decomposition (additive) has four main steps:

  1. Computing a trend-cycle component (\(T_t\)) using a moving average. A moving average is a technique for smoothing time series data by averaging the values of neighboring points. This helps to remove short-term fluctuations and highlight longer-term trends.

  2. Computing a series without the trend-cycle component (\(y_t - T_t\)).

  3. Estimating the seasonal component (\(S_t\)) by averaging the values from the detrended series for the season.

  4. Computing the remainder component (\(R_t\)) by subtracting the trend-cycle and seasonal components from the original series. \(R_t = y_t - T_t - S_t\)

We can use the classical_decomposition function inside of the model() function to decompose our time series data.

injuries |> 
  model(classical_decomposition(Count)) |>
  components() |>
  autoplot() +
  labs(title = "Classical Additive Decomposition of Monthly MLB Injury Counts")

Classical multiplicative decomposition works similarly to the additive decomposition, but with a few key differences.

  • The detrended series is computed as \(y_t / T_t\) instead of \(y_t - T_t\).

  • The seasonal component is estimated as \(S_t = y_t / T_t\) instead of \(y_t - T_t\).

  • The remainder component is computed as \(R_t = y_t / (T_t \times S_t)\) instead of \(y_t - T_t - S_t\).

NOTE: Ideally, after doing a decomposition, the remainder component should be white noise.

Classical multiplicative decomposition can be used by setting the type argument to "multiplicative" in the classical_decomposition() function.

STL Decomposition

Seasonal and Trend decomposition using Loess (STL) is a more advanced method for decomposing time series data. It is more flexible than classical decomposition and can handle any type of seasonality, not just monthly or quarterly. It also allows the user to control the length of the smoothing window for the trend-cycle. Lastly, it is more robust to outliers so that they do not affect the trend and seasonal estimates as much.

Below is an example of how to use the STL() function to decompose the time series data.

injuries |> 
  model(STL(Count ~ trend(window = 21)+ 
            season(window = "periodic"),
            robust = TRUE)) |>
  components() |>
  autoplot() +
  labs(title = "STL Additive Decomposition of Monthly MLB Injury Counts")

TIP: The window argument in the trend() function controls the length of the smoothing window for the trend-cycle. The larger the window, the smoother the trend. This value should always be an odd number so that a central point can be used. trend(window = 21) is a common choice for monthly data. This relatively large window size helps to prevent the trend from being influenced by short-term fluctuations in just a single year, but rather capture long-term trends and cycles.

TIP: The window argument in the season() function controls how many years the seasonal component should be estimated over. The default value is 11. When the seasonal window is set to periodic season(window = "periodic"), it is the equivalent setting the window to all of the data. When periodic is used, the seasonal component is assumed to be the same each year. The seasonal window argument should always be an odd number or “periodic”.

Exercise 3: Changing Decomposition Types
  1. Create a classical multiplicative decomposition of the monthly MLB injury counts. How do the components differ from the additive decomposition? Which decomposition method’s remainder component looks more like white noise (classical additive or classical multiplicative)?

  2. Create an STL decomposition of the monthly MLB injury counts with a shorter length for the trend smoothing window. How does the decomposition change with a shorter trend smoothing window? Particularly, how does the trend component change?

NOTE: You can actually forecast with decomposition as well. If you’d like to learn more about this click here

Basic Forecasting Methods

Now that we’ve explored the data and decomposed it into its components to better understand the patterns in the data, we’ll move on to forecasting. There are many methods for forecasting time series data, but we’ll focus on some basic methods in this section.

The Mean Method

An extremely simple method for forecasting is the mean method. This method forecasts the next observation as the average of all the observations in the training data. This method will produce a flat forecast that is equal to the mean of the training data. The mean method is useful when the data doesn’t have a trend or seasonality.

The Naive Method

The naive method is another simple forecasting method. It forecasts the next observation as the value of the last observation in the training data. This method will produce a flat forecast that is equal to the last observation in the training data. The naive method is useful when the data appears to be random.

Seasonal Naive Method

The seasonal naive method is similar to the naive method, but it forecasts the next observation as the value from the same season in the previous year. This method will produce a repeating pattern of forecasts that are equal to the observations from the same season in the previous year. (Basically forever repeating the last year’s pattern). The seasonal naive method is useful when the data has a strong seasonal pattern but no trend.

Drift Method

The drift method is a simple forecasting method that assumes a linear trend in the data. It forecasts the next observation as the value of the last observation plus the average change between observations. This method will produce a forecast that will continue on a linear trend from the first observation and through the last observation in the training data. The drift method is useful when the data has a linear trend but no seasonality.

Exercise 4: Basic Forecasting Methods

  1. Which of the above time series plots would be best forecasted using the mean method?

  2. Which of the above time series plots would be best forecasted using the seasonal naive method?

  3. Which of the above time series plots would be best forecasted using the drift method?

Use the following plots of the monthly MLB injury counts and forecasts to answer the following questions.

  1. After looking at the forecasts from the mean, naive, seasonal naive, and drift methods for the MLB injury data, which method appears to be the best forecast?

  2. Which ones appear to be the worst forecasts?

  3. What is one major issue seen in all of the forecasts in regards to their prediction intervals?

Evaluating Forecasting Models

Residuals

Checking the residuals of a model is one of the most effective ways to see how well the model is performing. Residuals are the differences between the observed values and the values predicted by the model. \(e_t = y_t - \hat{y}_t\)

It is important to note that when we are talking about residuals in this module that we are referring to the innovation residuals. Most of the time innovation residuals are the same as regular residuals, such as with our seasonal naive model. Innovation residuals are the residuals that are left over after accounting for changes made to the data such as transformations or differencing. These residuals are the ones that are used to check the model assumptions and to evaluate the model’s performance.

There are 3 main things to look at when evaluating residuals:

  • Do the residuals appear to be white noise? Remember that white noise has a mean of 0, constant variance, and shows no obvious patterns. This can be checked by looking at a time plot of the residuals.

  • Are the residuals normally distributed? This can be checked by looking at a histogram of the residuals or by using a normal probability plot.

  • Are the residuals uncorrelated? This can be checked by looking at the ACF plot of the residuals. There are also statistical tests that can be used to check for autocorrelation in the residuals such as the Ljung-Box test. We can use the Box.test() function in R to perform this test.

The formula for the test statistic for a Ljung-Box test is:

\[Q^{*} = T(T+2) \sum_{k=1}^{l} \frac{r^2_k}{T-k}\]

where:

  • \(T\) is the number of observations
  • \(l\) is the max number of lags
  • \(r_k\) is the sample autocorrelation at lag \(k\)

The null hypothesis for the Ljung-Box test is that the data is not distinguishable from white noise. If the p-value is less than the significance level, we reject the null hypothesis and conclude that the data is autocorrelated.

Thankfully there is a very easy way to check all of these at once in R using the gg_tsresiduals() function.

Exercise 5: Residuals Analysis

Below is code that will create a time plot, histogram, and ACF plot of the residuals from the seasonal naive method.

snaive_mod <- injuries |>
  model(SNAIVE(Count ~ lag('year')))
  
snaive_mod |> 
  gg_tsresiduals()

  1. Do the residuals appear to be white noise?

  2. What stands out about the time plot of the residuals?

  3. Does the histogram of the residuals appear to be normally distributed?

  4. Are the residuals uncorrelated? What lag(s) show the most significant autocorrelation and what could this mean for the model?

Testing and Training for Point Estimate Evaluations

If you want to evaluate the point estimates of a model, you can use a testing and training split. This involves training the model on the first part of the data and then testing the model on the second part of the data. This allows you to see how well the model can forecast future observations.

For this example, we will split the data into a training set that contains the first 75% of the data and a testing set that contains the last 25% of the data. This means we will train the models on the data from January 2001 to December 2017 and test the models on the data from January 2018 to December 2023.

TIP: 75-25 is a common split for training and testing data, but you can use any split that makes sense for your data. Generally the more data you have, the less percentage you need for testing. Other common splits are 70-30 or 80-20.

training <- injuries |> filter(year(Month) < 2018)
testing <- injuries |> filter(year(Month) >= 2018)

Now that we have our training and testing data, we can fit the models to the training data and then forecast the testing data.

injury_fit <- training |> 
  model(mean = MEAN(Count),
        naive = NAIVE(Count),
        snaive = SNAIVE(Count ~ lag('year')),
        drift = RW(Count ~ drift()))

injury_forecasts <- injury_fit |> 
  forecast(new_data = testing)

TIP: You can fit multiple models at once by using the model() function as seen in the code to the left. The values to the left of the = are the names we are giving to the models and the values to the right of the = are the models we are fitting to the data.

TIP: When using the forecast() function, you can specify the new data you want to forecast by using the new_data argument. In this case, we are forecasting for the testing data.

Let’s visualize the forecasts from the training data and compare them to the testing data.

injury_forecasts |>
  autoplot(injuries, level = NULL) +
  labs(title = "Forecasting Methods for Monthly MLB Injury Counts")+
  guides(color = guide_legend(title = "Forecast"))

The seasonal naive method certainly appears to be the best forecast.

We can also evaluate these models using the accuracy() function. This function calculates a variety of accuracy measures for the forecasts, including the mean absolute error, root mean squared error, mean absolute percentage error, and more.


The mean absolute error (MAE) is the average of the absolute errors between the forecasts and the actual values. It is a measure of the average magnitude of the errors in the forecasts. We want this value to be as close to 0 as possible.

The formula for the mean absolute error is: \[\text{MAE} = \frac{1}{n} \sum_{i=1}^{n} | y_i - \hat{y}_i |\] where \(y_i\) is the actual value and \(\hat{y}_i\) is the forecasted value.


The root mean squared error (RMSE) is the square root of the average of the squared errors between the forecasts and the actual values. It is a measure of the standard deviation of the errors in the forecasts. We want this value to be as close to 0 as possible.

The formula for the root mean squared error is: \[\text{RMSE} = \sqrt{\frac{1}{n} \sum_{i=1}^{n} (y_i - \hat{y}_i)^2}\] Where \(y_i\) is the actual value and \(\hat{y}_i\) is the forecasted value.


The mean absolute percentage error (MAPE) is the average of the absolute percentage errors between the forecasts and the actual values. It is a measure of the accuracy of the forecasts. We want this value to be as close to 0 as possible.

The formula for the mean absolute percentage error is: \[\text{MAPE} = \frac{100}{n} \sum_{i=1}^{n} \left| \frac{y_i - \hat{y}_i}{y_i} \right|\] where \(y_i\) is the actual value and \(\hat{y}_i\) is the forecasted value.


The code below displays the accuracy measures for the forecasts.

accuracy(injury_forecasts, testing)
# A tibble: 4 × 10
  .model .type    ME  RMSE   MAE   MPE  MAPE  MASE RMSSE  ACF1
  <chr>  <chr> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl>
1 drift  Test   86.0 122.   86.0   Inf   Inf   NaN   NaN 0.698
2 mean   Test   36.2  93.9  79.2  -Inf   Inf   NaN   NaN 0.698
3 naive  Test   86.0 122.   86.0   100   100   NaN   NaN 0.698
4 snaive Test   14.3  58.4  34.5  -Inf   Inf   NaN   NaN 0.424

This confirms that the seasonal naive method is the best forecast, as it has the lowest MAE and RMSE values. The MAPE is shown at -Inf to Inf because some of the actual values are 0, which causes the percentage error to be infinite.

Forecasting

Since the seasonal naive method appears to be the best for forecasting based on our training and testing data, we will use this method to forecast the future number of Tommy John surgeries in MLB players.

final_fit <- injuries |> 
  model(SNAIVE(Count ~ lag('year')))

This model is then used to forecast for the next 3 years.

final_forecasts <- final_fit |> 
  forecast(h = 36)

TIP: The forecast() function can be used to forecast for a specific number of periods into the future by using the h argument. In this case, we are forecasting for 36 months (3 years). This could also be written as forecast(h = "3 years").

The code below visualizes the forecasted values.

final_forecasts |>
  autoplot(injuries) +
  labs(title = "Forecasting MLB Injuries")+
  guides(color = guide_legend(title = "Forecast"))

Exercise 6: Forecasting MLB Injuries
  1. Based on the prediction intervals, are we more precise about the forecasted number of injuries projected in 2024 or 2026? What is the general trend seen in the prediction intervals?

  2. What is the forecasted number of players placed on the IL in July 2024.

Use this code to extract the forecasted values for July 2024:

final_forecasts |> filter(Month == yearmonth("2024-07"))
  1. What is the 95% prediction interval for the forecasted number of players placed on the IL in July 2024?

The hilo function can be used to extract the upper and lower bounds of the prediction interval.

final_forecasts |> filter(Month == yearmonth("2024-07")) |> hilo()
  1. If you were told that 112 players were placed on the IL in July 2024, would that be surprising? Would you consider your forecast made 7 months earlier to be successful?

More Practice

We have covered a lot of ground in this module. To practice what you have learned, try the exercises below on your own. Each of these exercises will use data regarding the number of Tommy John surgeries performed by year on MLB and MiLB players. The data can be downloaded below:

The image below shows that ulnar collateral ligament in the elbow of a pitcher. This ligament is often injured due to the extreme torque placed on the elbow during the pitching motion as it whips the arm forward. Ulnar Collateral Ligament

Image Source: R.J. Molenaars, CC BY-SA 4.0, via Wikimedia Commons

Tommy John surgery is a procedure in which a damaged ulnar collateral ligament in the elbow is replaced with a tendon from elsewhere in the body, usually the forearm. The surgery is named after the first baseball player to undergo the procedure, Tommy John, in 1974. The surgery is common among baseball pitchers and other athletes who put a lot of stress on their elbows. As Tommy John surgery has become more successful, the number of surgeries performed has increased, with some players even having the surgery multiple times.

Challenge 1: Visualizing the Data
  1. Load the Tommy John data into R and create a time series plot of the number of Tommy John surgeries performed by year.
tj_surgeries <- read_csv("tj_surgeries_mlb_milb.csv") |> 
  as_tsibble(index = year)
  1. Does the data exhibit trend?

  2. Does the data exhibit seasonality?

  3. Does the data exhibit non-seasonal trend cycles?

  4. Look at the ACF plot for the number of Tommy John surgeries. Does there appear to be any autocorrelation? If so, at what lags?

You can pipe the data into the gg_tsdisplay() function to create a time series plot and ACF plot at the same time.

  1. If there is no seasonality in data is there any reason to do a classical decomposition?

Injured Earlier?

Not only have the number of Tommy John surgeries increased over time for MLB and MiLB players, but high school and college players are also undergoing the surgery at increasing rates. Early career injuries are becoming more common, with some players undergoing the surgery long before they reach the professional level.

Nathan Eovaldi, pitcher for the 2018 World Series Champion Red Sox and 2023 World Series champions Texas Rangers, underwent Tommy John surgery in 2007 in high school. He was drafted by the Los Angeles Dodgers in 2008 and made his MLB debut in 2011. Eovaldi has since had a successful career, despite undergoing a second Tommy John surgery in 2016.

Nathan Eovaldi

Image Source: Arturo Pardavila III on Flickr (Original version) UCinternational (Crop), CC BY 2.0, via Wikimedia Commons)

Challenge 2: Simple Forecasting Methods

Use the following data to split the Tommy John surgeries data into training and testing sets. The training set will contain data from 1974 to 2011, and the testing set will contain data from 2012 to 2023.

training <- tj_surgeries |> filter(year <= 2011)
testing <- tj_surgeries |> filter(year > 2011)
  1. Fit a mean, naive, and drift model to the training data.
  1. Plot the forecasts for the mean, naive, and drift models for the testing data along with the actual testing data. Which appears to be the best forecast? Did the models overestimate, underestimate, or accurately predict the number of Tommy John surgeries?

Pipe your models into forecast(new_data = testing) and then pipe that into autoplot(testing). Set level = NULL in the autoplot() function to remove the prediction intervals from the plot.

  1. Check the accuracy of the models on the testing data. Which model has the lowest MAE, RMSE, and MAPE values?

Pipe your models into forecast(new_data = testing) and then pipe that into accuracy(testing) to get the accuracy measures.

  1. Fit the best type of model to the entire Tommy John dataset and forecast the number of Tommy John surgeries for the next 3 years. Plot the forecast along with the actual data. Does this forecast seem reasonable?
  1. What are the forecasted values for the number of Tommy John surgeries in 2024? 2025? 2026?
  1. What is the 95% prediction interval for the number of Tommy John surgeries in 2024? 2025? 2026?

Conclusion

In this module you have learned how to visualize time series data. You have also learned how the number of injuries in the MLB have changed over time. These injuries have major consequences for teams and the league as a whole, harming both performance and bottom lines. We have looked at how the injuries in the MLB change based on the month of the season, seeing higher rates shortly after the season starts and at the end of the season. This tells us that teams need to be cautious with player health particularly when they are not fully warmed up at the beginning of the season and when they are fatigued at the end of the season.

Major outliers in the number of injuries were noted during and directly after the COVID-19 pandemic. These most likely can be credited to being shut down for a while and then suddenly resuming play. Lastly, you looked at different basic forecasting methods - mean, drift, naive, and seasonal naive- to forecast the number of injuries in the MLB. Performance metrics and residual plots were used to determine the best model. A seasonal naive model was the best of the simple models we looked at, since it was able to capture the seasonality in the monthly data. However, this model still falls a little short because it captures no long-term trends that may be occurring in the data.

Additionally, you have learned about increases in Tommy John surgeries among baseball players and have now performed a simple time series analysis all by yourself.

The topics covered in this module are just the beginning of time series analysis and meant to help you get started working with time-related data. There are many more complex and accurate forecasting methods that can be used for advanced time series forecasting such as ARIMA, TSLM, ETS, Prophet, and Neural Networks that you should look into if you are interested if learning more.

References

  • Hyndman, R.J., & Athanasopoulos, G. (2021). Forecasting: Principles and Practice (3rd ed.). OTexts: Melbourne, Australia. Online