Source code for skopt.sampler.halton

"""
Inspired by https://github.com/jonathf/chaospy/blob/master/chaospy/
distributions/sampler/sequences/halton.py
"""
import numpy as np
from .base import InitialPointGenerator
from ..space import Space
from sklearn.utils import check_random_state


[docs]class Halton(InitialPointGenerator): """Creates `Halton` sequence samples. In statistics, Halton sequences are sequences used to generate points in space for numerical methods such as Monte Carlo simulations. Although these sequences are deterministic, they are of low discrepancy, that is, appear to be random for many purposes. They were first introduced in 1960 and are an example of a quasi-random number sequence. They generalise the one-dimensional van der Corput sequences. For ``dim == 1`` the sequence falls back to Van Der Corput sequence. Parameters ---------- min_skip : int minimum skipped seed number. When `min_skip != max_skip` a random number is picked. max_skip : int maximum skipped seed number. When `min_skip != max_skip` a random number is picked. primes : tuple, default=None The (non-)prime base to calculate values along each axis. If empty or None, growing prime values starting from 2 will be used. """
[docs] def __init__(self, min_skip=-1, max_skip=-1, primes=None): self.primes = primes self.min_skip = min_skip self.max_skip = max_skip
[docs] def generate(self, dimensions, n_samples, random_state=None): """Creates samples from Halton set. Parameters ---------- dimensions : list, shape (n_dims,) List of search space dimensions. Each search dimension can be defined either as - a `(lower_bound, upper_bound)` tuple (for `Real` or `Integer` dimensions), - a `(lower_bound, upper_bound, "prior")` tuple (for `Real` dimensions), - as a list of categories (for `Categorical` dimensions), or - an instance of a `Dimension` object (`Real`, `Integer` or `Categorical`). n_samples : int The order of the Halton sequence. Defines the number of samples. random_state : int, RandomState instance, or None (default) Set random state to something other than None for reproducible results. Returns ------- np.array, shape=(n_dim, n_samples) Halton set """ rng = check_random_state(random_state) if self.primes is None: primes = [] else: primes = list(self.primes) space = Space(dimensions) n_dim = space.n_dims transformer = space.get_transformer() space.set_transformer("normalize") if len(primes) < n_dim: prime_order = 10 * n_dim while len(primes) < n_dim: primes = _create_primes(prime_order) prime_order *= 2 primes = primes[:n_dim] assert len(primes) == n_dim, "not enough primes" if self.min_skip < 0 and self.max_skip < 0: skip = max(primes) elif self.min_skip == self.max_skip: skip = self.min_skip elif self.min_skip < 0 or self.max_skip < 0: skip = np.max(self.min_skip, self.max_skip) else: skip = rng.randint(self.min_skip, self.max_skip) out = np.empty((n_dim, n_samples)) indices = [idx + skip for idx in range(n_samples)] for dim_ in range(n_dim): out[dim_] = _van_der_corput_samples( indices, number_base=primes[dim_]) out = space.inverse_transform(np.transpose(out)) space.set_transformer(transformer) return out
def _van_der_corput_samples(idx, number_base=2): """ Create `Van Der Corput` low discrepancy sequence samples. A van der Corput sequence is an example of the simplest one-dimensional low-discrepancy sequence over the unit interval; it was first described in 1935 by the Dutch mathematician J. G. van der Corput. It is constructed by reversing the base-n representation of the sequence of natural numbers (1, 2, 3, ...). In practice, use Halton sequence instead of Van Der Corput, as it is the same, but generalized to work in multiple dimensions. Parameters ---------- idx (int, numpy.ndarray): The index of the sequence. If array is provided, all values in array is returned. number_base : int The numerical base from where to create the samples from. Returns ------- float, numpy.ndarray Van der Corput samples. """ assert number_base > 1 idx = np.asarray(idx).flatten() + 1 out = np.zeros(len(idx), dtype=float) base = float(number_base) active = np.ones(len(idx), dtype=bool) while np.any(active): out[active] += (idx[active] % number_base)/base idx //= number_base base *= number_base active = idx > 0 return out def _create_primes(threshold): """ Generate prime values using sieve of Eratosthenes method. Parameters ---------- threshold : int The upper bound for the size of the prime values. Returns ------ List All primes from 2 and up to ``threshold``. """ if threshold == 2: return [2] elif threshold < 2: return [] numbers = list(range(3, threshold+1, 2)) root_of_threshold = threshold ** 0.5 half = int((threshold+1)/2-1) idx = 0 counter = 3 while counter <= root_of_threshold: if numbers[idx]: idy = int((counter*counter-3)/2) numbers[idy] = 0 while idy < half: numbers[idy] = 0 idy += counter idx += 1 counter = 2*idx+3 return [2] + [number for number in numbers if number]