set.seed(1) knitr::opts_chunk$set( collapse = TRUE, comment = "#>" )
The main goal of the black box optimization toolkit (bbotk
) is to provide a common framework for optimization for other packages.
Therefore bbotk
includes the following R6 classes that can be used in a variety of optimization scenarios.
Optimizer
: Objects of this class allow you to optimize an object of the class OptimInstance
.OptimInstance
: Defines the optimization problem, consisting of an Objective
, the search_space
and a Terminator
.
All evaluations on the OptimInstance
will be automatically stored in its own Archive
.Objective
: Objects of this class contain the objective function.
The class ensures that the objective function is called in the right way and defines, whether the function should be minimized or maximized.Terminator
: Objects of this class control the termination of the optimization independent of the optimizer.As bbotk
also includes some basic optimizers and can be used on its own.
The registered optimizers can be queried as follows:
library(bbotk) opts()
This Vignette will show you how to use the bbotk
-classes to solve a simple optimization problem.
Furthermore, you will learn how to
Objective
.OptimInstance
search_space
.Archive
of evaluated function calls.bbotk
to optimize a functionIn the following we will use bbotk
to minimize this function:
fun = function(xs) { c(y = - (xs[[1]] - 2)^2 - (xs[[2]] + 3)^2 + 10) }
First we need to wrap fun
inside an Objective
object.
For functions that expect a list as input we can use the ObjectiveRFun
class.
Additionally, we need to specify the domain, i.e. the space of x-values that the function accepts as an input.
Optionally, we can define the co-domain, i.e. the output space of our objective function.
This is only necessary if we want to deviate from the default which would define the output to be named y and be minimized.
Such spaces are defined using the package paradox
.
library(paradox) domain = ps( x1 = p_dbl(-10, 10), x2 = p_dbl(-5, 5) ) codomain = ps( y = p_dbl(tags = "maximize") ) obfun = ObjectiveRFun$new( fun = fun, domain = domain, codomain = codomain, properties = "deterministic" # i.e. the result always returns the same result for the same input. )
In the next step we decide when the optimization should stop. We can list all available terminators as follows:
trms()
The termination should stop, when it takes longer then 10 seconds or when 20 evaluations are reached.
terminators = list( evals = trm("evals", n_evals = 20), run_time = trm("run_time") ) terminators
We have to correct the default of secs=30
by setting the values
in the param_set
of the terminator.
terminators$run_time$param_set$values$secs = 10
We have created Terminator
objects for both of our criteria.
To combine them we use the combo Terminator
.
term_combo = TerminatorCombo$new(terminators = terminators)
Before we finally start the optimization, we have to create an OptimInstance
that contains also the Objective
and the Terminator
.
instance = OptimInstanceSingleCrit$new(objective = obfun, terminator = term_combo) instance
Note, that OptimInstance(SingleCrit/MultiCrit)$new()
also has an optional search_space
argument.
It can be used if the search_space
is only a subset of obfun$domain
or if you want to apply transformations.
More on that later.
Finally, we have to define an Optimizer
.
As we have seen above, that we can call opts()
to list all available optimizers.
We opt for evolutionary optimizer, from the GenSA
package.
optimizer = opt("gensa") optimizer
To start the optimization we have to call the Optimizer
on the OptimInstance
.
optimizer$optimize(instance)
Note, that we did not specify the termination inside the optimizer.
bbotk
generally sets the termination of the optimizers to never terminate and instead breaks the code internally as soon as a termination criterion is fulfilled.
The results can be queried from the OptimInstance
.
# result as a data.table instance$result # result as a list that can be passed to the Objective instance$result_x_domain # result outcome instance$result_y
You can also access the whole history of evaluated points.
as.data.table(instance$archive)
If the domain of the Objective
is complex, it is often necessary to define a simpler search space that can be handled by the Optimizer
and to define a transformation that transforms the value suggested by the optimizer to a value of the domain of the Objective
.
Reasons for transformations can be:
x3
the change from 0.1
to 0.2
has a similar effect as the change of 100
to 200
.In the following we will look at an example for 2.)
We want to construct a box with the maximal volume, with the restriction that h+w+d = 1
.
For simplicity we define our problem as a minimization problem.
fun_volume = function(xs) { c(y = - (xs$h * xs$w * xs$d)) } domain = ps( h = p_dbl(lower = 0), w = p_dbl(lower = 0), d = p_dbl(lower = 0) ) obj = ObjectiveRFun$new( fun = fun_volume, domain = domain )
We notice, that our optimization problem has three parameters but due to the restriction it only the degree of freedom 2.
Therefore we only need to optimize two parameters and calculate h
, w
, d
accordingly.
search_space = ps( h = p_dbl(lower = 0, upper = 1), w = p_dbl(lower = 0, upper = 1), .extra_trafo = function(x, param_set){ x = unlist(x) x["d"] = 2 - sum(x) # model d in dependence of h, w x = x/sum(x) # ensure that h+w+d = 1 as.list(x) } )
Instead of the domain of the Objective
we now use our constructed search_space
that includes the trafo
for the OptimInstance
.
inst = OptimInstanceSingleCrit$new( objective = obj, search_space = search_space, terminator = trm("evals", n_evals = 30) ) optimizer = opt("gensa") lg = lgr::get_logger("bbotk")$set_threshold("warn") # turn off console output optimizer$optimize(inst) lg = lgr::get_logger("bbotk")$set_threshold("info") # turn on console output
The optimizer only saw the search space during optimization and returns the following result:
inst$result_x_search_space
Internally, the OptimInstance
transformed these values to the domain so that the result for the Objective
looks as follows:
inst$result_x_domain obj$eval(inst$result_x_domain)
The following is just meant for advanced readers.
If you want to evaluate the function outside of the optimization but have the result stored in the Archive
you can do so by resetting the termination and call $eval_batch()
.
library(data.table) inst$terminator = trm("none") xvals = data.table(h = c(0.6666, 0.6667), w = c(0.6666, 0.6667)) inst$eval_batch(xdt = xvals) tail(as.data.table(instance$archive))
However, this does not update the result.
You could set the result by calling inst$assign_result()
but this should be handled by the optimizer.
Another way to get the best point is the following:
inst$archive$best()
Note, that this method just looks for the best outcome and returns the according row from the archive. For stochastic optimization problems this is overly optimistic and leads to biased results. For this reason the optimizer can use advanced methods to determine the result and set it itself.
# TODO: Write the following sections: ## Implementing your own Objective ### Storing extra output in the archive ## Implementing your own Optimizer ### Storing extra tuner information in the archive ## Implement your own Terminator # TODO: Fix intro after more sections are added
Any scripts or data that you put into this service are public.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.