Note
Go to the end to download the full example code.
How to Implement Your Callback with OptunaHub
This recipe shows how to implement and register your own callback with OptunaHub.
Callbacks are used when you want to insert custom processing after each trial completes. Typical use cases include:
Uploading the current best result to an external server (e.g., W&B, MLflow).
Sending a notification when a new best value is found.
Stopping the study early based on custom criteria by calling
stop().
A callback is simply a callable with the following signature:
def callback(study: optuna.Study, trial: optuna.trial.FrozenTrial) -> None:
...
Optuna calls each registered callback once per trial, after the objective function
returns and the trial state has been recorded.
At that point Study already reflects the updated best value / best params,
and trial is a FrozenTrial (immutable).
The simplest way to implement a callback is to define a plain function:
from __future__ import annotations
import optuna
def my_callback(study: optuna.Study, trial: optuna.trial.FrozenTrial) -> None:
print(f"Trial {trial.number} finished.")
print(f" params : {trial.params}")
print(f" value : {trial.value}")
if trial.state == optuna.trial.TrialState.COMPLETE:
print(f" best_value : {study.best_value}")
If your callback needs to hold internal state (e.g., a connection to an external service),
you can implement it as a class with a __call__ method instead.
class MyCallback:
"""A callback that prints trial information after each completed trial.
Args:
verbose: If ``True``, also print the full ``params`` dict.
"""
def __init__(self, verbose: bool = True) -> None:
self._verbose = verbose
def __call__(self, study: optuna.Study, trial: optuna.trial.FrozenTrial) -> None:
# This method is called after every trial regardless of its state.
if trial.state != optuna.trial.TrialState.COMPLETE:
return
print(f"Trial {trial.number} finished.")
if self._verbose:
print(f" params : {trial.params}")
print(f" value : {trial.value}")
print(f" best_value : {study.best_value}")
The callback is passed to optimize() via the callbacks argument.
Multiple callbacks can be specified as a list; they are called in order after each trial.
def objective(trial: optuna.trial.Trial) -> float:
x = trial.suggest_float("x", -10, 10)
y = trial.suggest_int("y", -5, 5)
return x**2 + y**2
Run the study with a plain function callback.
study = optuna.create_study()
study.optimize(objective, n_trials=5, callbacks=[my_callback])
Trial 0 finished.
params : {'x': -9.413079018475148, 'y': -4}
value : 104.60605660805706
best_value : 104.60605660805706
Trial 1 finished.
params : {'x': -3.40735819594995, 'y': -4}
value : 27.6100898755073
best_value : 27.6100898755073
Trial 2 finished.
params : {'x': -9.403377486629953, 'y': -3}
value : 97.42350815605906
best_value : 27.6100898755073
Trial 3 finished.
params : {'x': 4.631015144709751, 'y': -1}
value : 22.44630127053108
best_value : 22.44630127053108
Trial 4 finished.
params : {'x': 4.088819500763437, 'y': -2}
value : 20.71844490982336
best_value : 20.71844490982336
Run another study with the class-based callback.
study = optuna.create_study()
study.optimize(objective, n_trials=5, callbacks=[MyCallback(verbose=True)])
Trial 0 finished.
params : {'x': -9.74151780566059, 'y': 3}
value : 103.89716915800233
best_value : 103.89716915800233
Trial 1 finished.
params : {'x': 9.443854544958278, 'y': 1}
value : 90.18638866632912
best_value : 90.18638866632912
Trial 2 finished.
params : {'x': 3.8774885815355216, 'y': -1}
value : 16.03491769993835
best_value : 16.03491769993835
Trial 3 finished.
params : {'x': -4.160774232024327, 'y': -4}
value : 33.31204220987763
best_value : 16.03491769993835
Trial 4 finished.
params : {'x': 1.789424881139471, 'y': -1}
value : 4.20204140524101
best_value : 4.20204140524101
After implementing your own callback, you can register it with OptunaHub.
See How to Register Your Package with OptunaHub for how to register your callback with OptunaHub.
The category name to use when placing your package in the registry is callbacks:
Total running time of the script: (0 minutes 0.008 seconds)