use crate::core::math::Scalar;
use crate::core::problem::EvalCounts;
use crate::core::state::{CountsMirror, RhoState, State};
pub struct NewuoaState<V, F = f64> {
pub(crate) param: V,
pub(crate) cost: Option<F>,
pub(crate) rho: F,
pub(crate) best_param: Option<V>,
pub(crate) best_cost: F,
pub(crate) best_iter: u64,
pub(crate) best_cost_evals: u64,
pub(crate) iter: u64,
pub(crate) cost_evals: u64,
}
impl<V, F: Scalar> NewuoaState<V, F> {
pub fn new(x0: V) -> Self {
Self {
param: x0,
cost: None,
rho: F::infinity(),
best_param: None,
best_cost: F::infinity(),
best_iter: 0,
best_cost_evals: 0,
iter: 0,
cost_evals: 0,
}
}
pub fn rho(&self) -> F {
self.rho
}
}
impl<V: Clone, F: Scalar> State for NewuoaState<V, F> {
type Param = V;
type Float = F;
fn iter(&self) -> u64 {
self.iter
}
fn increment_iter(&mut self) {
self.iter += 1;
}
fn cost_evals(&self) -> u64 {
self.cost_evals
}
fn param(&self) -> &V {
&self.param
}
fn cost(&self) -> F {
self.cost
.expect("NewuoaState::cost read before Solver::init evaluated the start point")
}
fn best_param(&self) -> &V {
self.best_param
.as_ref()
.expect("NewuoaState::best_param read before Solver::init populated it")
}
fn best_cost(&self) -> F {
self.best_cost
}
fn best_iter(&self) -> u64 {
self.best_iter
}
fn best_cost_evals(&self) -> u64 {
self.best_cost_evals
}
fn update_best(&mut self) {
if let Some(cost) = self.cost {
if self.best_param.is_none() || cost < self.best_cost {
self.best_param = Some(self.param.clone());
self.best_cost = cost;
self.best_iter = self.iter;
self.best_cost_evals = self.cost_evals;
}
}
}
fn reset_best(&mut self) {
self.best_param = None;
self.best_cost = F::infinity();
self.best_iter = 0;
self.best_cost_evals = 0;
}
}
impl<V: Clone, F: Scalar> RhoState for NewuoaState<V, F> {
fn rho(&self) -> F {
self.rho
}
}
impl<V, F> CountsMirror for NewuoaState<V, F>
where
NewuoaState<V, F>: State,
{
fn mirror(&mut self, delta: &EvalCounts) {
self.cost_evals = delta.total_work();
}
}