import numpy as np
from math import inf
from spotpython.fun.objectivefunctions import Analytical
from spotpython.utils.init import fun_control_init, design_control_init
from spotpython.hyperparameters.values import set_control_key_value
from spotpython.spot import Spot
import matplotlib.pyplot as plt
7 Introduction to spotpython
Surrogate model based optimization methods are common approaches in simulation and optimization. SPOT was developed because there is a great need for sound statistical analysis of simulation and optimization algorithms. SPOT includes methods for tuning based on classical regression and analysis of variance techniques. It presents tree-based models such as classification and regression trees and random forests as well as Bayesian optimization (Gaussian process models, also known as Kriging). Combinations of different meta-modeling approaches are possible. SPOT comes with a sophisticated surrogate model based optimization method, that can handle discrete and continuous inputs. Furthermore, any model implemented in scikit-learn
can be used out-of-the-box as a surrogate in spotpython
.
SPOT implements key techniques such as exploratory fitness landscape analysis and sensitivity analysis. It can be used to understand the performance of various algorithms, while simultaneously giving insights into their algorithmic behavior.
The spot
loop consists of the following steps:
- Init: Build initial design \(X\)
- Evaluate initial design on real objective \(f\): \(y = f(X)\)
- Build surrogate: \(S = S(X,y)\)
- Optimize on surrogate: \(X_0 = \text{optimize}(S)\)
- Evaluate on real objective: \(y_0 = f(X_0)\)
- Impute (Infill) new points: \(X = X \cup X_0\), \(y = y \cup y_0\).
- Goto 3.
Central Idea: Evaluation of the surrogate model S
is much cheaper (or / and much faster) than running the real-world experiment \(f\). We start with a small example.
7.1 Example: Spot
and the Sphere Function
7.1.1 The Objective Function: Sphere
The spotpython
package provides several classes of objective functions. We will use an analytical objective function, i.e., a function that can be described by a (closed) formula: \[
f(x) = x^2
\]
= Analytical().fun_sphere fun
We can apply the function fun
to input values and plot the result:
= np.linspace(-1,1,100).reshape(-1,1)
x = fun(x)
y
plt.figure()"k")
plt.plot(x, y, plt.show()
7.1.2 The Spot
Method as an Optimization Algorithm Using a Surrogate Model
We initialize the fun_control
dictionary. The fun_control
dictionary contains the parameters for the objective function. The fun_control
dictionary is passed to the Spot
method.
=fun_control_init(lower = np.array([-1]),
fun_control= np.array([1]))
upper = Spot(fun=fun,
spot_0 =fun_control)
fun_control spot_0.run()
Experiment saved to 000_exp.pkl
spotpython tuning: 4.960293502265715e-09 [#######---] 73.33%
spotpython tuning: 4.959666330154525e-09 [########--] 80.00%
spotpython tuning: 4.9571338392226926e-09 [#########-] 86.67%
spotpython tuning: 4.9571338392226926e-09 [#########-] 93.33%
spotpython tuning: 1.866838525968143e-10 [##########] 100.00% Done...
Experiment saved to 000_res.pkl
<spotpython.spot.spot.Spot at 0x1541a6cf0>
The method print_results()
prints the results, i.e., the best objective function value (“min y”) and the corresponding input value (“x0”).
spot_0.print_results()
min y: 1.866838525968143e-10
x0: 1.3663229947447064e-05
[['x0', np.float64(1.3663229947447064e-05)]]
To plot the search progress, the method plot_progress()
can be used. The parameter log_y
is used to plot the objective function values on a logarithmic scale.
=True) spot_0.plot_progress(log_y
If the dimension of the input space is one, the method plot_model()
can be used to visualize the model and the underlying objective function values.
spot_0.plot_model()
7.2 Spot
Parameters: fun_evals
, init_size
and show_models
We will modify three parameters:
- The number of function evaluations (
fun_evals
) will be set to10
(instead of 15, which is the default value) in thefun_control
dictionary. - The parameter
show_models
, which visualizes the search process for each single iteration for 1-dim functions, in thefun_control
dictionary. - The size of the initial design (
init_size
) in thedesign_control
dictionary.
The full list of the Spot
parameters is shown in code reference on GitHub, see Spot.
=fun_control_init(lower = np.array([-1]),
fun_control= np.array([1]),
upper = 10,
fun_evals = True)
show_models = design_control_init(init_size=9)
design_control = Spot(fun=fun,
spot_1 =fun_control,
fun_control=design_control)
design_control spot_1.run()
Experiment saved to 000_exp.pkl
spotpython tuning: 9.632846333472212e-09 [##########] 100.00% Done...
Experiment saved to 000_res.pkl
7.3 Print the Results
spot_1.print_results()
min y: 9.632846333472212e-09
x0: -9.814706482351988e-05
[['x0', np.float64(-9.814706482351988e-05)]]
7.4 Show the Progress
spot_1.plot_progress()
7.5 Visualizing the Optimization and Hyperparameter Tuning Process with TensorBoard
spotpython
supports the visualization of the hyperparameter tuning process with TensorBoard. The following example shows how to use TensorBoard with spotpython
.
First, we define an “PREFIX” to identify the hyperparameter tuning process. The PREFIX is used to create a directory for the TensorBoard files.
= fun_control_init(
fun_control = "01",
PREFIX = np.array([-1]),
lower = np.array([2]),
upper =100,
fun_evals=True,
TENSORBOARD_CLEAN=True)
tensorboard_log= design_control_init(init_size=5) design_control
Moving TENSORBOARD_PATH: runs/ to TENSORBOARD_PATH_OLD: runs_OLD/runs_2025_01_21_20_07_44_0
Created spot_tensorboard_path: runs/spot_logs/01_maans08_2025-01-21_20-07-44 for SummaryWriter()
Since the tensorboard_log
is True
, spotpython
will log the optimization process in the TensorBoard files. The argument TENSORBOARD_CLEAN=True
will move the TensorBoard files from the previous run to a backup folder, so that TensorBoard files from previous runs are not overwritten and a clean start in the runs
folder is guaranteed.
= Spot(fun=fun,
spot_tuner =fun_control,
fun_control=design_control)
design_control
spot_tuner.run() spot_tuner.print_results()
Experiment saved to 01_exp.pkl
spotpython tuning: 2.487613964476317e-05 [#---------] 6.00%
spotpython tuning: 7.85064251023503e-07 [#---------] 7.00%
spotpython tuning: 7.210284620467494e-07 [#---------] 8.00%
spotpython tuning: 4.191441813872346e-07 [#---------] 9.00%
spotpython tuning: 7.4015092835768465e-09 [#---------] 10.00%
spotpython tuning: 7.4015092835768465e-09 [#---------] 11.00%
spotpython tuning: 7.4015092835768465e-09 [#---------] 12.00%
spotpython tuning: 7.4015092835768465e-09 [#---------] 13.00%
spotpython tuning: 7.4015092835768465e-09 [#---------] 14.00%
spotpython tuning: 7.4015092835768465e-09 [##--------] 15.00%
spotpython tuning: 7.4015092835768465e-09 [##--------] 16.00%
spotpython tuning: 7.4015092835768465e-09 [##--------] 17.00%
spotpython tuning: 7.4015092835768465e-09 [##--------] 18.00%
spotpython tuning: 7.4015092835768465e-09 [##--------] 19.00%
spotpython tuning: 7.4015092835768465e-09 [##--------] 20.00%
spotpython tuning: 6.704802934300911e-10 [##--------] 21.00%
spotpython tuning: 6.704802934300911e-10 [##--------] 22.00%
spotpython tuning: 6.668629799831143e-10 [##--------] 23.00%
spotpython tuning: 6.668629799831143e-10 [##--------] 24.00%
spotpython tuning: 6.499816856638441e-10 [##--------] 25.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 26.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 27.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 28.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 29.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 30.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 31.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 32.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 33.00%
spotpython tuning: 6.499816856638441e-10 [###-------] 34.00%
spotpython tuning: 6.499816856638441e-10 [####------] 35.00%
spotpython tuning: 6.499816856638441e-10 [####------] 36.00%
spotpython tuning: 6.499816856638441e-10 [####------] 37.00%
spotpython tuning: 6.499816856638441e-10 [####------] 38.00%
spotpython tuning: 6.499816856638441e-10 [####------] 39.00%
spotpython tuning: 6.499816856638441e-10 [####------] 40.00%
spotpython tuning: 6.499816856638441e-10 [####------] 41.00%
spotpython tuning: 6.499816856638441e-10 [####------] 42.00%
spotpython tuning: 6.499816856638441e-10 [####------] 43.00%
spotpython tuning: 6.499816856638441e-10 [####------] 44.00%
spotpython tuning: 6.499816856638441e-10 [####------] 45.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 46.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 47.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 48.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 49.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 50.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 51.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 52.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 53.00%
spotpython tuning: 6.499816856638441e-10 [#####-----] 54.00%
spotpython tuning: 6.499816856638441e-10 [######----] 55.00%
spotpython tuning: 6.499816856638441e-10 [######----] 56.00%
spotpython tuning: 6.499816856638441e-10 [######----] 57.00%
spotpython tuning: 6.499816856638441e-10 [######----] 58.00%
spotpython tuning: 6.499816856638441e-10 [######----] 59.00%
spotpython tuning: 6.499816856638441e-10 [######----] 60.00%
spotpython tuning: 6.499816856638441e-10 [######----] 61.00%
spotpython tuning: 6.499816856638441e-10 [######----] 62.00%
spotpython tuning: 6.499816856638441e-10 [######----] 63.00%
spotpython tuning: 6.499816856638441e-10 [######----] 64.00%
spotpython tuning: 6.499816856638441e-10 [######----] 65.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 66.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 67.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 68.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 69.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 70.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 71.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 72.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 73.00%
spotpython tuning: 6.499816856638441e-10 [#######---] 74.00%
spotpython tuning: 6.499816856638441e-10 [########--] 75.00%
spotpython tuning: 6.499816856638441e-10 [########--] 76.00%
spotpython tuning: 6.499816856638441e-10 [########--] 77.00%
spotpython tuning: 6.499816856638441e-10 [########--] 78.00%
spotpython tuning: 6.499816856638441e-10 [########--] 79.00%
spotpython tuning: 6.499816856638441e-10 [########--] 80.00%
spotpython tuning: 6.499816856638441e-10 [########--] 81.00%
spotpython tuning: 6.499816856638441e-10 [########--] 82.00%
spotpython tuning: 6.499816856638441e-10 [########--] 83.00%
spotpython tuning: 6.499816856638441e-10 [########--] 84.00%
spotpython tuning: 6.499816856638441e-10 [########--] 85.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 86.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 87.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 88.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 89.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 90.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 91.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 92.00%
spotpython tuning: 6.499816856638441e-10 [#########-] 93.00%
spotpython tuning: 4.991706867647581e-10 [#########-] 94.00%
spotpython tuning: 4.991706867647581e-10 [##########] 95.00%
spotpython tuning: 4.991706867647581e-10 [##########] 96.00%
spotpython tuning: 4.991706867647581e-10 [##########] 97.00%
spotpython tuning: 4.991706867647581e-10 [##########] 98.00%
spotpython tuning: 4.991706867647581e-10 [##########] 99.00%
spotpython tuning: 4.3763009967401825e-10 [##########] 100.00% Done...
Experiment saved to 01_res.pkl
min y: 4.3763009967401825e-10
x0: 2.0919610409231293e-05
[['x0', np.float64(2.0919610409231293e-05)]]
Now we can start TensorBoard in the background. The TensorBoard process will read the TensorBoard files and visualize the hyperparameter tuning process. From the terminal, we can start TensorBoard with the following command:
tensorboard --logdir="./runs"
logdir
is the directory where the TensorBoard files are stored. In our case, the TensorBoard files are stored in the directory ./runs
.
TensorBoard will start a web server on port 6006. We can access the TensorBoard web server with the following URL:
http://localhost:6006/
The first TensorBoard visualization shows the objective function values plotted against the wall time. The wall time is the time that has passed since the start of the hyperparameter tuning process. The five initial design points are shown in the upper left region of the plot. The line visualizes the optimization process.
The second TensorBoard visualization shows the input values, i.e., \(x_0\), plotted against the wall time.
The third TensorBoard plot illustrates how spotpython
can be used as a microscope for the internal mechanisms of the surrogate-based optimization process. Here, one important parameter, the learning rate \(\theta\) of the Kriging surrogate is plotted against the number of optimization steps.
7.6 Jupyter Notebook
- The Jupyter-Notebook of this lecture is available on GitHub in the Hyperparameter-Tuning-Cookbook Repository