OptimLib is a lightweight C++ library of numerical optimization methods for nonlinear functions.
Features:
- C++11 library of local and global optimization algorithms, as well as root finding techniques.
- Derivative-free optimization using advanced, parallelized metaheuristics.
- Constrained optimization routines can handle simple box constraints, or systems of nonlinear constraints.
- Built on the Armadillo C++ linear algebra library for fast and efficient matrix-based computation.
- OpenMP-accelerated library, and straightforward linking with parallelized BLAS libraries such as OpenBLAS.
- Released under a permissive, non-GPL license.
The library is actively maintained, and is still being extended. A list of algorithms includes:
- Broyden's Method
- Newton's method, BFGS, and L-BFGS
- Nonlinear Conjugate Gradient
- Nelder-Mead
- Differential Evolution (DE)
- Particle Swarm Optimization (PSO)
OptimLib functions generally have the following structure:
algorithm(<initial-to-output values>, <objective function>, <objective function data>)
In order:
- A writable vector of initial values that defines the starting point of the algorithm, and will contain the solution vector upon successful completion of the algorithm.
- The 'objective function' is the function to be minimized, or zeroed-out in the case of Broyden's method.
- The final input is optional: it is any object that contains additional parameters required to evaluate the objective function.
For example, the BFGS algorithm is called using
bool bfgs(arma::vec& init_out_vals, std::function<double (const arma::vec& vals_inp, arma::vec* grad_out, void* opt_data)> opt_objfn, void* opt_data);
The library is installed using the standard ./configure && make
method:
# clone optim into the current directory
git clone https://github.com/kthohr/optim ./optim
# build and install
cd ./optim
./configure -i "/usr/local" -p
make
make install
The final command will install OptimLib into /usr/local
.
There are several configuration options available (see ./configure -h
):
-c
a coverage build (used with Codecov)-d
a 'development' build-g
a debugging build (optimization flags set to-O0 -g
)-h
print help-i
install path; default: the build directory-m
specify the BLAS and Lapack libraries to link against; for example,-m "-lopenblas"
or-m "-framework Accelerate"
-o
compiler optimization options; defaults to-O3 -march=native -ffp-contract=fast -flto -DARMA_NO_DEBUG
-p
enable OpenMP parallelization features (recommended)
OptimLib is built on the Armadillo C++ linear algebra library. The configure
script will search for Armadillo files in the usual places: /usr/include
, /usr/local/include
, /opt/include
, /opt/local/include
. If the Armadillo header files are installed elsewhere, set the following environment variable before running configure
:
export ARMA_INCLUDE_PATH=/path/to/armadillo
Otherwise the build script will download the required files from the Armadillo GitHub repository.
To illustrate OptimLib at work, consider the problem of finding the global minimum of the Ackley function:
This is a well-known test function with many local minima. Newton-type methods (such as BFGS) are sensitive to the choice of initial values, and will perform rather poorly here. As such, we will employ a global search method; in this case: Differential Evolution.
Code:
#include "optim.hpp"
//
// Ackley function
double ackley_fn(const arma::vec& vals_inp, arma::vec* grad_out, void* opt_data)
{
const double x = vals_inp(0);
const double y = vals_inp(1);
const double pi = arma::datum::pi;
double obj_val = -20*std::exp( -0.2*std::sqrt(0.5*(x*x + y*y)) ) - std::exp( 0.5*(std::cos(2*pi*x) + std::cos(2*pi*y)) ) + 22.718282L;
//
return obj_val;
}
int main()
{
// initial values:
arma::vec x = arma::ones(2,1) + 1.0; // (2,2)
//
std::chrono::time_point<std::chrono::system_clock> start = std::chrono::system_clock::now();
bool success = optim::de(x,ackley_fn,nullptr);
std::chrono::time_point<std::chrono::system_clock> end = std::chrono::system_clock::now();
std::chrono::duration<double> elapsed_seconds = end-start;
if (success) {
std::cout << "de: Ackley test completed successfully.\n"
<< "elapsed time: " << elapsed_seconds.count() << "s\n";
} else {
std::cout << "de: Ackley test completed unsuccessfully." << std::endl;
}
arma::cout << "\nde: solution to Ackley test:\n" << x << arma::endl;
return 0;
}
Output:
de: Ackley test completed successfully.
elapsed time: 0.028167s
de: solution to Ackley test:
-1.2702e-17
-3.8432e-16
On a standard laptop, OptimLib will compute a solution to within machine precision in a fraction of a second.
Check the /tests
directory for additional examples, and http://www.kthohr.com/optimlib.html for a detailed description of each algorithm.
Keith O'Hara
Apache Version 2