mathru 0.16.2

Fundamental algorithms for scientific computing in Rust
Documentation
use crate::{
    algebra::{
        abstr::Real,
        linear::{matrix::General, vector::vector::Vector},
    },
    optimization::{Optim, OptimResult},
};
#[cfg(feature = "serde")]
use serde::{Deserialize, Serialize};
use std::clone::Clone;
use std::marker::PhantomData;

/// Gauss-Newton method
#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
#[derive(Clone, Copy, Debug)]
pub struct GaussNewton<T> {
    iters: u64,
    __phantom: PhantomData<T>,
}

impl<T> GaussNewton<T> {
    pub fn new(iters: u64) -> GaussNewton<T> {
        GaussNewton {
            iters,
            __phantom: PhantomData,
        }
    }
}

impl<T> GaussNewton<T>
where
    T: Real,
{
    /// Minimize function func
    ///
    /// # Arguments
    ///
    /// * 'func0': Function to be minimized
    /// * 'x_0': Initial guess for the minimum
    ///
    /// # Return
    ///
    /// local minimum
    pub fn minimize<F>(&self, func: &F, x_0: &Vector<T>) -> Result<OptimResult<Vector<T>>, ()>
    where
        F: Optim<T>,
    {
        let mut x_n: Vector<T> = x_0.clone();

        for _i in 0..self.iters {
            let jacobian_x_n: General<T> = func.jacobian(&x_n);
            let f_x_n: Vector<T> = func.eval(&x_n);
            let pinv = jacobian_x_n.pinv()?;
            let delta_x_n: Vector<T> = pinv * f_x_n;
            x_n -= delta_x_n;
        }

        Ok(OptimResult::new(x_n))
    }
}