use std::fmt::Debug;
use std::cmp::PartialOrd;
use std::default::Default;
use std::ops::{Index, IndexMut};
use rand;
use rand::distributions::{IndependentSample, Range};
use ndarray::Array1;
pub trait ArgminParameter
: Clone
+ Default
+ Debug
+ Index<usize, Output = <Self as ArgminParameter>::Element>
+ IndexMut<usize> {
type Element: PartialOrd + Clone;
fn modify(&self) -> (Self, usize);
fn random(&Self, &Self) -> Self;
}
macro_rules! random_vec_iter {
($type:ty) => {
fn random(lower_bound: &$type, upper_bound: &$type) -> $type {
let mut rng = rand::thread_rng();
let out: $type = lower_bound
.iter()
.zip(upper_bound.iter())
.map(|(l, u)| {
if l >= u {
panic!("Parameter: lower_bound must be lower than upper_bound.");
}
let range = Range::new(*l, *u);
range.ind_sample(&mut rng)
})
.collect();
out
}
}
}
macro_rules! modify_one_parameter {
() => {
fn modify(&self) -> (Self, usize) {
let pos = Range::new(0, self.len());
let range = Range::new(-1.0, 1.0);
let mut rng = rand::thread_rng();
let mut param = self.clone();
let idx = pos.ind_sample(&mut rng);
param[idx] = self[idx] + range.ind_sample(&mut rng);
(param, idx)
}
}
}
macro_rules! implement_argmin_parameter {
($param:ty, $element:ty) => {
impl ArgminParameter for $param {
type Element = $element;
modify_one_parameter!();
random_vec_iter!($param);
}
}
}
implement_argmin_parameter!(Vec<f64>, f64);
implement_argmin_parameter!(Vec<f32>, f32);
implement_argmin_parameter!(Array1<f64>, f64);
implement_argmin_parameter!(Array1<f32>, f32);