Introduction
To use the code in this article, you will need to install the following packages: kernlab, modeldata, themis, and tidymodels.
Many of the examples for model tuning focus on grid search. For that method, all the candidate tuning parameter combinations are defined prior to evaluation. Alternatively, iterative search can be used to analyze the existing tuning parameter results and then predict which tuning parameters to try next.
There are a variety of methods for iterative search and the focus in this article is on Bayesian optimization. For more information on this method, these resources might be helpful:
-
Practical bayesian optimization of machine learning algorithms (2012). J Snoek, H Larochelle, and RP Adams. Advances in neural information.
-
A Tutorial on Bayesian Optimization for Machine Learning (2018). R Adams.
-
Gaussian Processes for Machine Learning (2006). C E Rasmussen and C Williams.
Cell segmenting revisited
To demonstrate this approach to tuning models, let’s return to the cell segmentation data from the Getting Started article on resampling:
library(tidymodels)
library(modeldata)
# Load data
data(cells)
set.seed(2369)
tr_te_split <- initial_split(cells %>% select(-case), prop = 3/4)
cell_train <- training(tr_te_split)
cell_test <- testing(tr_te_split)
set.seed(1697)
folds <- vfold_cv(cell_train, v = 10)
The tuning scheme
Since the predictors are highly correlated, we can used a recipe to convert the original predictors to principal component scores. There is also slight class imbalance in these data; about 64% of the data are poorly segmented. To mitigate this, the data will be down-sampled at the end of the pre-processing so that the number of poorly and well segmented cells occur with equal frequency. We can use a recipe for all this pre-processing, but the number of principal components will need to be tuned so that we have enough (but not too many) representations of the data.
library(themis)
cell_pre_proc <-
recipe(class ~ ., data = cell_train) %>%
step_YeoJohnson(all_predictors()) %>%
step_normalize(all_predictors()) %>%
step_pca(all_predictors(), num_comp = tune()) %>%
step_downsample(class)
In this analysis, we will use a support vector machine to model the data. Let’s use a radial basis function (RBF) kernel and tune its main parameter ($\sigma$). Additionally, the main SVM parameter, the cost value, also needs optimization.
svm_mod <-
svm_rbf(mode = "classification", cost = tune(), rbf_sigma = tune()) %>%
set_engine("kernlab")
These two objects (the recipe and model) will be combined into a single object via the workflow()
function from the workflows package; this object will be used in the optimization process.
svm_wflow <-
workflow() %>%
add_model(svm_mod) %>%
add_recipe(cell_pre_proc)
From this object, we can derive information about what parameters are slated to be tuned. A parameter set is derived by:
svm_set <- extract_parameter_set_dials(svm_wflow)
svm_set
#> Collection of 3 parameters for tuning
#>
#> identifier type object
#> cost cost nparam[+]
#> rbf_sigma rbf_sigma nparam[+]
#> num_comp num_comp nparam[+]
The default range for the number of PCA components is rather small for this data set. A member of the parameter set can be modified using the update()
function. Let’s constrain the search to one to twenty components by updating the num_comp
parameter. Additionally, the lower bound of this parameter is set to zero which specifies that the original predictor set should also be evaluated (i.e., with no PCA step at all):
svm_set <-
svm_set %>%
update(num_comp = num_comp(c(0L, 20L)))
Sequential tuning
Bayesian optimization is a sequential method that uses a model to predict new candidate parameters for assessment. When scoring potential parameter value, the mean and variance of performance are predicted. The strategy used to define how these two statistical quantities are used is defined by an acquisition function.
For example, one approach for scoring new candidates is to use a confidence bound. Suppose accuracy is being optimized. For a metric that we want to maximize, a lower confidence bound can be used. The multiplier on the standard error (denoted as \(\kappa\)
) is a value that can be used to make trade-offs between exploration and exploitation.
-
Exploration means that the search will consider candidates in untested space.
-
Exploitation focuses in areas where the previous best results occurred.
The variance predicted by the Bayesian model is mostly spatial variation; the value will be large for candidate values that are not close to values that have already been evaluated. If the standard error multiplier is high, the search process will be more likely to avoid areas without candidate values in the vicinity.
We’ll use another acquisition function, expected improvement, that determines which candidates are likely to be helpful relative to the current best results. This is the default acquisition function. More information on these functions can be found in the package vignette for acquisition functions.
set.seed(12)
search_res <-
svm_wflow %>%
tune_bayes(
resamples = folds,
# To use non-default parameter ranges
param_info = svm_set,
# Generate five at semi-random to start
initial = 5,
iter = 50,
# How to measure performance?
metrics = metric_set(roc_auc),
control = control_bayes(no_improve = 30, verbose = TRUE)
)
#>
#> ❯ Generating a set of 5 initial parameter results
#> ✓ Initialization complete
#>
#> Optimizing roc_auc using the expected improvement
#>
#> ── Iteration 1 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8761 (@iter 0)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00386, rbf_sigma=0.00513, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.879 (+/-0.0109)
#>
#> ── Iteration 2 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.879 (@iter 1)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.0338, rbf_sigma=0.00332, num_comp=13
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8736 (+/-0.0127)
#>
#> ── Iteration 3 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.879 (@iter 1)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.275, rbf_sigma=0.00304, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.8792 (+/-0.0118)
#>
#> ── Iteration 4 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8792 (@iter 3)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=9.56, rbf_sigma=0.00426, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.8869 (+/-0.00942)
#>
#> ── Iteration 5 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8869 (@iter 4)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=26, rbf_sigma=0.00617, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.8894 (+/-0.00914)
#>
#> ── Iteration 6 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8894 (@iter 5)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=28.5, rbf_sigma=0.0054, num_comp=2
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.7961 (+/-0.00994)
#>
#> ── Iteration 7 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8894 (@iter 5)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=19.7, rbf_sigma=0.802, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.7641 (+/-0.00967)
#>
#> ── Iteration 8 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8894 (@iter 5)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=9.84, rbf_sigma=0.00434, num_comp=17
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8863 (+/-0.00992)
#>
#> ── Iteration 9 ───────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8894 (@iter 5)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=12.8, rbf_sigma=0.0138, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8821 (+/-0.00847)
#>
#> ── Iteration 10 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8894 (@iter 5)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=20.2, rbf_sigma=0.00842, num_comp=18
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8889 (+/-0.00942)
#>
#> ── Iteration 11 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8894 (@iter 5)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=19.5, rbf_sigma=0.0082, num_comp=15
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.8963 (+/-0.00915)
#>
#> ── Iteration 12 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8963 (@iter 11)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=17.1, rbf_sigma=0.0096, num_comp=14
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.8986 (+/-0.00935)
#>
#> ── Iteration 13 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=1.91, rbf_sigma=1.21e-10, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.3458 (+/-0.115)
#>
#> ── Iteration 14 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=23.2, rbf_sigma=0.0127, num_comp=14
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8932 (+/-0.00991)
#>
#> ── Iteration 15 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=23.1, rbf_sigma=0.00885, num_comp=12
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8958 (+/-0.00939)
#>
#> ── Iteration 16 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=19.9, rbf_sigma=0.00783, num_comp=13
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8955 (+/-0.00977)
#>
#> ── Iteration 17 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=24.4, rbf_sigma=0.0241, num_comp=14
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8758 (+/-0.0127)
#>
#> ── Iteration 18 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=4.83, rbf_sigma=0.00892, num_comp=14
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8922 (+/-0.00973)
#>
#> ── Iteration 19 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=5.53, rbf_sigma=0.921, num_comp=0
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.7764 (+/-0.0178)
#>
#> ── Iteration 20 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=22.9, rbf_sigma=0.00957, num_comp=15
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8939 (+/-0.00948)
#>
#> ── Iteration 21 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00119, rbf_sigma=0.843, num_comp=9
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.3579 (+/-0.112)
#>
#> ── Iteration 22 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00135, rbf_sigma=0.00161, num_comp=18
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.3477 (+/-0.114)
#>
#> ── Iteration 23 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00883, rbf_sigma=0.0108, num_comp=16
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8773 (+/-0.0118)
#>
#> ── Iteration 24 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.0186, rbf_sigma=0.00653, num_comp=14
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8746 (+/-0.0124)
#>
#> ── Iteration 25 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00127, rbf_sigma=0.0133, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8813 (+/-0.0105)
#>
#> ── Iteration 26 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.8986 (@iter 12)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=4.41, rbf_sigma=0.0208, num_comp=9
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.9 (+/-0.00937)
#>
#> ── Iteration 27 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=5.55, rbf_sigma=0.153, num_comp=4
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8824 (+/-0.0107)
#>
#> ── Iteration 28 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=6.12, rbf_sigma=0.0497, num_comp=0
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8723 (+/-0.0106)
#>
#> ── Iteration 29 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=2.69, rbf_sigma=0.115, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8717 (+/-0.0096)
#>
#> ── Iteration 30 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.0037, rbf_sigma=0.00707, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8812 (+/-0.0111)
#>
#> ── Iteration 31 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00169, rbf_sigma=0.01, num_comp=5
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8758 (+/-0.0115)
#>
#> ── Iteration 32 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=27.3, rbf_sigma=0.185, num_comp=12
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.831 (+/-0.0112)
#>
#> ── Iteration 33 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=2.23, rbf_sigma=0.286, num_comp=13
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8524 (+/-0.00999)
#>
#> ── Iteration 34 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9 (@iter 26)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=7.56, rbf_sigma=0.0162, num_comp=9
#> i Estimating performance
#> ✓ Estimating performance
#> ♥ Newest results: roc_auc=0.9001 (+/-0.0097)
#>
#> ── Iteration 35 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=4.49, rbf_sigma=0.0377, num_comp=12
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8905 (+/-0.0102)
#>
#> ── Iteration 36 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=30.9, rbf_sigma=0.00156, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8836 (+/-0.0098)
#>
#> ── Iteration 37 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=13.5, rbf_sigma=0.000277, num_comp=17
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8743 (+/-0.012)
#>
#> ── Iteration 38 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=5.38, rbf_sigma=0.000549, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8794 (+/-0.0114)
#>
#> ── Iteration 39 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=27.9, rbf_sigma=3.41e-05, num_comp=13
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8727 (+/-0.013)
#>
#> ── Iteration 40 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=4.23, rbf_sigma=5.66e-05, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8768 (+/-0.0111)
#>
#> ── Iteration 41 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=7.35, rbf_sigma=6.16e-05, num_comp=2
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.7963 (+/-0.00995)
#>
#> ── Iteration 42 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=29.6, rbf_sigma=7.84e-05, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.878 (+/-0.0118)
#>
#> ── Iteration 43 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=9.14, rbf_sigma=1.29e-05, num_comp=20
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8774 (+/-0.0115)
#>
#> ── Iteration 44 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=17.3, rbf_sigma=3.31e-05, num_comp=19
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8766 (+/-0.0119)
#>
#> ── Iteration 45 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=27.7, rbf_sigma=4.99e-06, num_comp=0
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8708 (+/-0.0125)
#>
#> ── Iteration 46 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=3.32, rbf_sigma=8.21e-06, num_comp=3
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8663 (+/-0.0131)
#>
#> ── Iteration 47 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=25.6, rbf_sigma=1.09e-05, num_comp=5
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8742 (+/-0.0114)
#>
#> ── Iteration 48 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.804, rbf_sigma=2.45e-06, num_comp=18
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.3479 (+/-0.114)
#>
#> ── Iteration 49 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=5.26, rbf_sigma=2.08e-05, num_comp=11
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.8736 (+/-0.0124)
#>
#> ── Iteration 50 ──────────────────────────────────────────────────────
#>
#> i Current best: roc_auc=0.9001 (@iter 34)
#> i Gaussian process model
#> ✓ Gaussian process model
#> i Generating 5000 candidates
#> i Predicted candidates
#> i cost=0.00125, rbf_sigma=0.0301, num_comp=1
#> i Estimating performance
#> ✓ Estimating performance
#> ⓧ Newest results: roc_auc=0.7725 (+/-0.0106)
The resulting tibble is a stacked set of rows of the rsample object with an additional column for the iteration number:
search_res
#> # Tuning results
#> # 10-fold cross-validation
#> # A tibble: 510 × 5
#> splits id .metrics .notes .iter
#> <list> <chr> <list> <list> <int>
#> 1 <split [1362/152]> Fold01 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 2 <split [1362/152]> Fold02 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 3 <split [1362/152]> Fold03 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 4 <split [1362/152]> Fold04 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 5 <split [1363/151]> Fold05 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 6 <split [1363/151]> Fold06 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 7 <split [1363/151]> Fold07 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 8 <split [1363/151]> Fold08 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 9 <split [1363/151]> Fold09 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> 10 <split [1363/151]> Fold10 <tibble [5 × 7]> <tibble [0 × 3]> 0
#> # … with 500 more rows
As with grid search, we can summarize the results over resamples:
estimates <-
collect_metrics(search_res) %>%
arrange(.iter)
estimates
#> # A tibble: 55 × 10
#> cost rbf_sigma num_comp .metric .estimator mean n std_err .config
#> <dbl> <dbl> <int> <chr> <chr> <dbl> <int> <dbl> <chr>
#> 1 0.00207 1.56e-5 10 roc_auc binary 0.344 10 0.114 Prepro…
#> 2 0.348 4.43e-2 1 roc_auc binary 0.773 10 0.0106 Prepro…
#> 3 15.5 1.28e-7 20 roc_auc binary 0.346 10 0.115 Prepro…
#> 4 1.45 2.04e-3 15 roc_auc binary 0.876 10 0.0122 Prepro…
#> 5 0.0304 6.41e-9 5 roc_auc binary 0.344 10 0.114 Prepro…
#> 6 0.00386 5.13e-3 19 roc_auc binary 0.879 10 0.0109 Iter1
#> 7 0.0338 3.32e-3 13 roc_auc binary 0.874 10 0.0127 Iter2
#> 8 0.275 3.04e-3 20 roc_auc binary 0.879 10 0.0118 Iter3
#> 9 9.56 4.26e-3 20 roc_auc binary 0.887 10 0.00942 Iter4
#> 10 26.0 6.17e-3 19 roc_auc binary 0.889 10 0.00914 Iter5
#> # … with 45 more rows, and 1 more variable: .iter <int>
The best performance of the initial set of candidate values was AUC = 0.876
. The best results were achieved at iteration 34 with a corresponding AUC value of 0.9. The five best results are:
show_best(search_res, metric = "roc_auc")
#> # A tibble: 5 × 10
#> cost rbf_sigma num_comp .metric .estimator mean n std_err .config .iter
#> <dbl> <dbl> <int> <chr> <chr> <dbl> <int> <dbl> <chr> <int>
#> 1 7.56 0.0162 9 roc_auc binary 0.900 10 0.00970 Iter34 34
#> 2 4.41 0.0208 9 roc_auc binary 0.900 10 0.00937 Iter26 26
#> 3 17.1 0.00960 14 roc_auc binary 0.899 10 0.00935 Iter12 12
#> 4 19.5 0.00820 15 roc_auc binary 0.896 10 0.00915 Iter11 11
#> 5 23.1 0.00885 12 roc_auc binary 0.896 10 0.00939 Iter15 15
A plot of the search iterations can be created via:
autoplot(search_res, type = "performance")
There are many parameter combinations have roughly equivalent results.
How did the parameters change over iterations?
autoplot(search_res, type = "parameters") +
labs(x = "Iterations", y = NULL)
Session information
#> ─ Session info ─────────────────────────────────────────────────────
#> setting value
#> version R version 4.2.1 (2022-06-23)
#> os macOS Big Sur ... 10.16
#> system x86_64, darwin17.0
#> ui X11
#> language (EN)
#> collate en_US.UTF-8
#> ctype en_US.UTF-8
#> tz America/Los_Angeles
#> date 2022-12-07
#> pandoc 2.19.2 @ /Applications/RStudio.app/Contents/MacOS/quarto/bin/tools/ (via rmarkdown)
#>
#> ─ Packages ─────────────────────────────────────────────────────────
#> package * version date (UTC) lib source
#> broom * 1.0.1 2022-08-29 [1] CRAN (R 4.2.0)
#> dials * 1.1.0 2022-11-04 [1] CRAN (R 4.2.0)
#> dplyr * 1.0.10 2022-09-01 [1] CRAN (R 4.2.0)
#> ggplot2 * 3.4.0 2022-11-04 [1] CRAN (R 4.2.0)
#> infer * 1.0.4 2022-12-02 [1] CRAN (R 4.2.1)
#> kernlab * 0.9-31 2022-06-09 [1] CRAN (R 4.2.0)
#> modeldata * 1.0.1 2022-09-06 [1] CRAN (R 4.2.0)
#> parsnip * 1.0.3 2022-11-11 [1] CRAN (R 4.2.0)
#> purrr * 0.3.5 2022-10-06 [1] CRAN (R 4.2.0)
#> recipes * 1.0.3 2022-11-09 [1] CRAN (R 4.2.0)
#> rlang * 1.0.6 2022-09-24 [1] CRAN (R 4.2.0)
#> rsample * 1.1.1 2022-12-07 [1] CRAN (R 4.2.1)
#> themis * 1.0.0 2022-07-02 [1] CRAN (R 4.2.0)
#> tibble * 3.1.8 2022-07-22 [1] CRAN (R 4.2.0)
#> tidymodels * 1.0.0 2022-07-13 [1] CRAN (R 4.2.0)
#> tune * 1.0.1 2022-10-09 [1] CRAN (R 4.2.0)
#> workflows * 1.1.2 2022-11-16 [1] CRAN (R 4.2.0)
#> yardstick * 1.1.0 2022-09-07 [1] CRAN (R 4.2.0)
#>
#> [1] /Library/Frameworks/R.framework/Versions/4.2/Resources/library
#>
#> ────────────────────────────────────────────────────────────────────