use crate::fitness::FitnessFnWrapper;
use crate::genotypes::Range as RangeGenotype;
use crate::traits::ChromosomeT;
use std::borrow::Cow;
use std::fmt;
use std::fmt::Debug;
#[derive(Clone, Debug, PartialEq)]
#[cfg_attr(
feature = "serde",
derive(serde::Serialize, serde::Deserialize),
serde(bound(
serialize = "T: serde::Serialize",
deserialize = "T: serde::de::DeserializeOwned"
))
)]
pub struct Range<T: Sync + Send + Clone + Default + Debug> {
pub dna: Vec<RangeGenotype<T>>,
pub fitness: f64,
pub age: usize,
#[cfg_attr(feature = "serde", serde(skip, default))]
pub fitness_fn: FitnessFnWrapper<RangeGenotype<T>>,
}
impl<T: Sync + Send + Clone + Default + Debug> Default for Range<T> {
fn default() -> Self {
Self {
dna: Vec::new(),
fitness: f64::NAN,
age: 0,
fitness_fn: FitnessFnWrapper::default(),
}
}
}
impl<T: Sync + Send + Clone + Default + Debug> Range<T> {
pub fn new() -> Self {
Self {
dna: Vec::new(),
fitness: f64::NAN,
age: 0,
fitness_fn: FitnessFnWrapper::default(),
}
}
pub fn phenotype(&self) -> String {
self.dna
.iter()
.map(|gene| format!("{:?}", gene.value()))
.collect::<Vec<_>>()
.join(", ")
}
}
impl<T: Sync + Send + Clone + Default + Debug + 'static> ChromosomeT for Range<T> {
type Gene = RangeGenotype<T>;
fn dna(&self) -> &[Self::Gene] {
&self.dna
}
fn dna_mut(&mut self) -> &mut [Self::Gene] {
&mut self.dna
}
fn set_dna<'a>(&mut self, dna: Cow<'a, [Self::Gene]>) -> &mut Self {
self.dna = match dna {
Cow::Borrowed(slice) => slice.to_vec(),
Cow::Owned(vec) => vec,
};
self
}
fn set_fitness_fn<F>(&mut self, fitness_fn: F) -> &mut Self
where
F: Fn(&[RangeGenotype<T>]) -> f64 + Send + Sync + 'static,
{
self.fitness_fn = FitnessFnWrapper::new(fitness_fn);
self
}
fn calculate_fitness(&mut self) {
self.fitness = self.fitness_fn.call(&self.dna);
}
fn fitness(&self) -> f64 {
self.fitness
}
fn set_fitness(&mut self, fitness: f64) -> &mut Self {
self.fitness = fitness;
self
}
fn set_age(&mut self, age: usize) -> &mut Self {
self.age = age;
self
}
fn age(&self) -> usize {
self.age
}
}
impl<T: Sync + Send + Clone + Default + Debug + fmt::Display> fmt::Display for Range<T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "[{}] fitness={:.6}", self.phenotype(), self.fitness)
}
}