Saving/Resuming Study with RDB Backend

An RDB backend enables persistent experiments (i.e., to save and resume a study) as well as access to history of studies. In addition, we can run multi-node optimization tasks with this feature, which is described in 4. Easy Parallelization.

In this section, let’s try simple examples running on a local environment with SQLite DB.

Note

You can also utilize other RDB backends, e.g., PostgreSQL or MySQL, by setting the storage argument to the DB’s URL. Please refer to SQLAlchemy’s document for how to set up the URL.

New Study

We can create a persistent study by calling create_study() function as follows. An SQLite file example.db is automatically initialized with a new study record.

import logging
import sys

import optuna

# Add stream handler of stdout to show the messages
optuna.logging.get_logger("optuna").addHandler(logging.StreamHandler(sys.stdout))
study_name = "example-study"  # Unique identifier of the study.
storage_name = "sqlite:///{}.db".format(study_name)
study = optuna.create_study(study_name=study_name, storage=storage_name)
A new study created in RDB with name: example-study

To run a study, call optimize() method passing an objective function.

def objective(trial):
    x = trial.suggest_float("x", -10, 10)
    return (x - 2) ** 2


study.optimize(objective, n_trials=3)
Trial 0 finished with value: 0.0021270149031328593 and parameters: {'x': 2.046119571801274}. Best is trial 0 with value: 0.0021270149031328593.
Trial 1 finished with value: 34.10624945363487 and parameters: {'x': -3.840055603642389}. Best is trial 0 with value: 0.0021270149031328593.
Trial 2 finished with value: 91.22483326692495 and parameters: {'x': -7.551169209417501}. Best is trial 0 with value: 0.0021270149031328593.

Resume Study

To resume a study, instantiate a Study object passing the study name example-study and the DB URL sqlite:///example-study.db.

study = optuna.create_study(study_name=study_name, storage=storage_name, load_if_exists=True)
study.optimize(objective, n_trials=3)
Using an existing study with name 'example-study' instead of creating a new one.
Trial 3 finished with value: 8.572865964431005 and parameters: {'x': 4.927945690143689}. Best is trial 0 with value: 0.0021270149031328593.
Trial 4 finished with value: 25.02315228124821 and parameters: {'x': -3.002314692344756}. Best is trial 0 with value: 0.0021270149031328593.
Trial 5 finished with value: 1.605410043436606 and parameters: {'x': 0.7329522331669551}. Best is trial 0 with value: 0.0021270149031328593.

Note that the storage doesn’t store the state of the instance of samplers. When we resume a study with a sampler whose seed argument is specified for reproducibility, you need to restore the sampler with using pickle as follows:

import pickle

# Save the sampler with pickle to be loaded later.
with open("sampler.pkl", "wb") as fout:
    pickle.dump(study.sampler, fout)

restored_sampler = pickle.load(open("sampler.pkl", "rb"))
study = optuna.create_study(
    study_name=study_name, storage=storage_name, load_if_exists=True, sampler=restored_sampler
)
study.optimize(objective, n_trials=3)

Experimental History

We can access histories of studies and trials via the Study class. For example, we can get all trials of example-study as:

study = optuna.create_study(study_name=study_name, storage=storage_name, load_if_exists=True)
df = study.trials_dataframe(attrs=("number", "value", "params", "state"))
Using an existing study with name 'example-study' instead of creating a new one.

The method trials_dataframe() returns a pandas dataframe like:

print(df)
   number      value  params_x     state
0       0   0.002127  2.046120  COMPLETE
1       1  34.106249 -3.840056  COMPLETE
2       2  91.224833 -7.551169  COMPLETE
3       3   8.572866  4.927946  COMPLETE
4       4  25.023152 -3.002315  COMPLETE
5       5   1.605410  0.732952  COMPLETE

A Study object also provides properties such as trials, best_value, best_params (see also 1. Lightweight, versatile, and platform agnostic architecture).

print("Best params: ", study.best_params)
print("Best value: ", study.best_value)
print("Best Trial: ", study.best_trial)
print("Trials: ", study.trials)
Best params:  {'x': 2.046119571801274}
Best value:  0.0021270149031328593
Best Trial:  FrozenTrial(number=0, state=TrialState.COMPLETE, values=[0.0021270149031328593], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 153587), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 174131), params={'x': 2.046119571801274}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=1, value=None)
Trials:  [FrozenTrial(number=0, state=TrialState.COMPLETE, values=[0.0021270149031328593], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 153587), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 174131), params={'x': 2.046119571801274}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=1, value=None), FrozenTrial(number=1, state=TrialState.COMPLETE, values=[34.10624945363487], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 197354), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 213289), params={'x': -3.840055603642389}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=2, value=None), FrozenTrial(number=2, state=TrialState.COMPLETE, values=[91.22483326692495], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 229525), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 245370), params={'x': -7.551169209417501}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=3, value=None), FrozenTrial(number=3, state=TrialState.COMPLETE, values=[8.572865964431005], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 308042), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 325687), params={'x': 4.927945690143689}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=4, value=None), FrozenTrial(number=4, state=TrialState.COMPLETE, values=[25.02315228124821], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 346809), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 363029), params={'x': -3.002314692344756}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=5, value=None), FrozenTrial(number=5, state=TrialState.COMPLETE, values=[1.605410043436606], datetime_start=datetime.datetime(2022, 12, 22, 7, 12, 15, 378931), datetime_complete=datetime.datetime(2022, 12, 22, 7, 12, 15, 393661), params={'x': 0.7329522331669551}, user_attrs={}, system_attrs={}, intermediate_values={}, distributions={'x': FloatDistribution(high=10.0, log=False, low=-10.0, step=None)}, trial_id=6, value=None)]

Total running time of the script: ( 0 minutes 0.619 seconds)

Gallery generated by Sphinx-Gallery