Method PROTES (PRobabilistic Optimizer with TEnsor Sampling) for derivative-free optimization of the multidimensional arrays and discretized multivariate functions based on the tensor train (TT) format.
To use this package, please install manually first the python programming language of the version 3.8 or 3.9, then, the package can be installed via pip:
pip install protes==0.3.5To ensure version stability, we recommend working in a virtual environment, as described in the
workflow.md.
Let's say we want to find the minimum of a d = 10 dimensional array (tensor) that has n = 5 elements (indices) in each mode. Let an arbitrary tensor element y related to the d-dimensional multi-index i is defined by the function y = f(i) = sum(i). In this case, the optimizer may be launched in the following way:
import numpy as np
from protes import protes
f_batch = lambda I: np.sum(I, axis=1)
i_opt, y_opt = protes(f=f_batch, d=10, n=5, m=5.E+3, log=True)The function f_batch takes a set of multi-indices I (jax array having a size samples x d) and returns a list (or jax array or numpy array) of the corresponding tensor values; m is is the computational budget (the allowed number of requested tensor elements). Returned values i_opt (jax array) and y_opt (float) are the found multi-index and the value in this multi-index for the approximate optimum (minimum, since the flag is_max is not set) of the target tensor, respectively.
The input
Ican be easily converted to anumpyarray if needed (i.e.,import numpy as np; I = np.array(I)), so you you can use the ordinarynumpyinside the taget function and do not use thejaxat all.
Note that the code runs orders of magnitude faster if the tensor's mode size (
n) is the same for all modes, as shown in the example above. If you need to optimize a tensor with discriminating mode sizes, then you should use the slowprotes_generalfunction (i.e.,from protes import protes_general). In this case, instead of two parametersdandn, one parameternshould be passed, which is a list of the lengthdcorresponding to the mode sizes in each dimension (all other parameters for the functionprotes_generalare the same as for the functionprotes, which are detailed in the next section).
Please, see also the demo folder, which contains several examples of using the PROTES method for real tasks (a simple demo can be run in the console with a command python demo/demo_func.py and python demo/demo_qubo.py).
Mandatory arguments:
f(function) - the target functionf(I), where inputIis a 2D jax array of the shape[samples, d](dis a number of dimensions of the function's input andsamplesis a batch size of requested multi-indices). The function should return 1D list or jax array or numpy array of the length equals tosamples(the values of the target function for all provided multi-indices, i.e., the values of the optimized tensor).d(int) - number of tensor dimensions.n(int) - mode size for each tensor's dimension.
Optional arguments:
m(int) - the number of allowed requests to the objective function (the default value isNone). If this parameter is not set, then the optimization process will continue until the objective function returnsNoneinstead of a list of values.k(int) - the batch size for optimization (the default value is100).k_top(int) - number of selected candidates in the batch (it should be< k; the default value is10).k_gd(int) - number of gradient lifting iterations for each batch (the default value is1).lr(float): learning rate for gradient lifting iterations (the default value is5.E-2).r(int): TT-rank of the constructed probability TT-tensor (the default value is5).seed(int): parameter for jax random generator (the default value is0).is_max(bool): if flag is set, then maximization rather than minimization will be performed.log(bool): if flag is set, then the information about the progress of the algorithm will be printed after each improvement of the optimization result and at the end of the algorithm's work.info(dict): optional dictionary, which will be filled with reference information about the process of the algorithm operation.P(list): optional initial probability tensor in the TT-format (represented as a list of jax arrays, where all non-edge TT-cores are merged into one array; see the function_generate_initialinprotes.pyfor details). If this parameter is not set, then a random initial TT-tensor will be generated. Note that this tensor will be changed inplace.
-
You can use the outer cache for the values requested by the optimizer (that is, for each requested batch, check if any of the multi-indices have already been calculated), this can in some cases reduce the number of requests to the objective function. In this case, it is convenient not to set limits on the number of requests (parameter
m), but instead, when the budget is exhausted (including the cache), return theNonevalue from the target function (f), in which case the optimization algorithm will be automatically terminated. -
For a number of tasks, performance can be improved by switching to increased precision in the representation of numbers in
jax; for this, at the beginning of the script, you should specify the code:import jax jax.config.update('jax_enable_x64', True)
-
If there is a GPU, the
jaxoptimizer code will be automatically executed on it, however, the current version of the code works better on the CPU (also, we did not conduct thorough testing of the current version of the code on the GPU). To use CPU on a device with available GPU, you should specify the following code at the beginning of the executable script:import jax jax.config.update('jax_platform_name', 'cpu') jax.default_device(jax.devices('cpu')[0]);
If you find our approach and/or code useful in your research, please consider citing:
@article{batsheva2023protes,
author = {Batsheva, Anastasia and Chertkov, Andrei and Ryzhakov, Gleb and Oseledets, Ivan},
year = {2023},
title = {{PROTES}: {P}robabilistic optimization with tensor sampling},
journal = {arXiv preprint arXiv:2301.12162},
url = {https://arxiv.org/pdf/2301.12162.pdf}
}✭ 🚂 The stars that you give to PROTES, motivate us to develop faster and add new interesting features to the code 😃
If you plan to use this software for commercial purposes, please contact Ivan Oseledets for details. There are no restrictions on the use for scientific purposes when citing 😊