{tidymodels} Framework Example Model Construction`
Objectives: This notebook shows an example of a model being assessed and constructed using the framework proposed in notebook x_ModelingFrameworkAndTidymodelsReview. We’ll use the batted balls and home runs data that many of you have been working with for the last several class meetings. If you chose to use the FAA BirdStrikes and Engine Damage data, you should follow along and implement models with your data set.
The Data
As a reminder, the MLB batted balls data set was also associated with the park dimensions dataset. We’ll load both of those data sets below.
Rows: 13000 Columns: 25
── Column specification ────────────────────────────────────────────────────────
Delimiter: ","
chr (8): home_team, away_team, batter_team, batter_name, pitcher_name, bb_...
dbl (16): bip_id, batter_id, pitcher_id, is_batter_lefty, is_pitcher_lefty,...
date (1): game_date
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
Rows: 30 Columns: 9
── Column specification ────────────────────────────────────────────────────────
Delimiter: ","
chr (2): NAME, Cover
dbl (7): park, LF_Dim, CF_Dim, RF_Dim, LF_W, CF_W, RF_W
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.
Now that the data sets have been read into our notebook, we can view the first few rows of each as a reminder of what we are working with. The first two rows of the batted_balls data frame is below.
We can see that the two data frames share the park column. This means that we can join the two data sets together – adding information about the baseball stadium (including the field dimensions) to the batted_balls data frame. It is reasonable that knowing about the park dimensions will be helpful in identifying whether a batted ball is going to be a home run (hit over the outfield fence) or not.
In this notebook, we’ll show how to assess, fit and use a model to classify whether a batted ball will result in a home run (is_home_run) given features of the scenario leading to the batted ball. To keep things simple, we’ll use pitch_mph, launch_speed, launch_angle, and Cover as predictors. (Note: There are some really great predictors that we’ve left on the table here.)
Splitting Data and Creating Folds
We’ll start by splitting our data into training and test sets and create folds using our training data.
It looks like the nearest neighbor model outperforms the decision tree using ROC-AUC as the performance metric but not using accuracy. We’ll discuss more about model performance metrics at our next class meeting. For now, let’s fit the nearest neighbor model since both models had comparable accuracy but ROC-AUC was quite a bite better for nearest neighbors.
Fit, Assess, and Utilize Our “Best” Model
We’ll start by fitting our best model to our training data.
knn_fit <- knn_wf %>%fit(train)
Now that the model has been fitted, we’ll use our model to make predictions on our test observations. The performance metrics should align with our cross-validation estimate (if not, then we’ll have concerns which will prevent us from putting the model into production).
The model was just about as accurate as expected given the cross-validation results. Since it doesn’t look like our model is overfit, let’s use it to make a prediction on whether a new batted ball would result in a home run.
Looks like our model is very certain that won’t be a home run.
Summary
There it is – we’ve walked through a basic modeling process using the {tidymodels} framework. We’ll discuss this in greater detail and introduce more advanced techniques throughout MAT434.
Extension: Hyperparameters and Tuning
The workflow in the example outlined above is a standard basic approach to modeling. We built two vanilla models (keeping all default parameters) and applied them to our home run prediction problem. Taking a closer look at these models, how sure are we that we have the “best” model? We identified that the KNN classifier outperformed the decision tree classifier, but that is for a specific value of \(k\) (nearest neighbors) and for an unconstrained decision tree. Even though we didn’t override any settings, we made modeling choices by accepting the default parameters.
These parameters which must be set prior to your model seeing its training data are called model hyperparameters. Each class of model has different hyperparameters available to it, and you’ll be introduced to them throughout our course. Ideally, we’d like the optimal settings for these hyperparameters to be learned from our data, but how do we make that happen if they need to be set prior to the model seeing any training data?
Keep Calm and Cross-Validate
The {tidymodels} ecosystem contains a few specialized functions for using cross-validation to search over combinations of hyperparameter settings. I’ll introduce one here: tune_grid().
The process of using tune_grid() is similar to that of fit() or fit_resamples(). We’ll pass the tune_grid() function a model workflow, but rather than setting static hyperparameter values like tree_depth = 6, we’ll explicitly state that we want to tune that hyperparameter by setting tree_depth = tune().
We’ll attempt to optimize our nearest_neighbor() classifier over its neighbors hyperparameter, and our decision_tree() classifier over its tree_depth hyperparameter. We’ll create the parameter grids below. These are essentially tibbles (data frames) of options for each hyperparameter. An important consideration when tuning hyperparameters is that, in the case where we attempt to optimize a single model class over multiple hyperparameters, we’ll need to build a tibble containing all combinations of hyperparameters in each list. Be careful with this because it can result in cross-validating over many, many models.
Note. You can set the grid parameter of tune_grid() to a number which will create a “space filling” grid containing that number of hyperparameter combinations if you are attempting to tune across lots of hyperparameters.
Now that we have the grids, we’ll re-create our model workflows, setting the relevant hyperparameters to tune(). Then we’ll pipe these workflows into tune_grid() and read the results.
It looks like our best-performing nearest neighbor model is with 21 nearest neighbors, although 15 and 11 are not far behind.These models both sit at above 95.5% accuracy. It may be worth testing more options near 21 neighbors, since the closest options we considered were 15 and 41 – perhaps another value will be optimal.
Now let’s look at the results from our decision tree classifier.
Check this out! It looks like our optimized decision tree classifier (max_depth = 5) beats our best KNN classifier. The accuracy for this model is nearly 95.9% and has a standard deviation in fold accuracies comparable to the KNN models. Notice also that trees of depth 5, 8, 10, 12, 15, and 20 all perform similarly. This indicates that additional flexibility for our trees doesn’t improve model performance. The estimated model performance and standard error don’t change across these values, which indicates that there are some Pit of Success principles operating behind the scenes here, preventing our models from growing beyond a depth of 5.
Hyperparameter tuning helps improve your models on the margins, but sometimes these improvements can be quite significant.