src/euclidean.rs

Wed, 03 Sep 2025 09:16:03 -0500

author
Tuomo Valkonen <tuomov@iki.fi>
date
Wed, 03 Sep 2025 09:16:03 -0500
branch
dev
changeset 162
bea0c3841ced
parent 151
402d717bb5c0
child 164
fd9dba51afd3
permissions
-rw-r--r--

cow_owned etc.

/*!
Euclidean spaces.
*/

use crate::instance::Instance;
use crate::linops::{VectorSpace, AXPY};
use crate::norms::Reflexive;
use crate::types::*;

pub mod wrap;

// TODO: Euclidean & EuclideanMut
//
/// Space (type) with Euclidean and vector space structure
///
/// The type should implement vector space operations (addition, subtraction, scalar
/// multiplication and scalar division) along with their assignment versions, as well
/// as an inner product.
// TODO: remove F parameter, use VectorSpace::Field
pub trait Euclidean<F: Float = f64>:
    VectorSpace<Field = F, Owned = Self::OwnedEuclidean>
    + Reflexive<F, DualSpace = Self::OwnedEuclidean>
{
    type OwnedEuclidean: ClosedEuclidean<F>;

    // Inner product
    fn dot<I: Instance<Self>>(&self, other: I) -> F;

    /// Calculate the square of the 2-norm, $\frac{1}{2}\\|x\\|_2^2$, where `self` is $x$.
    ///
    /// This is not automatically implemented to avoid imposing
    /// `for <'a> &'a Self : Instance<Self>` trait bound bloat.
    fn norm2_squared(&self) -> F;

    /// Calculate the square of the 2-norm divided by 2, $\frac{1}{2}\\|x\\|_2^2$,
    /// where `self` is $x$.
    #[inline]
    fn norm2_squared_div2(&self) -> F {
        self.norm2_squared() / F::TWO
    }

    /// Calculate the 2-norm $‖x‖_2$, where `self` is $x$.
    #[inline]
    fn norm2(&self) -> F {
        self.norm2_squared().sqrt()
    }

    /// Calculate the 2-distance squared $\\|x-y\\|_2^2$, where `self` is $x$.
    fn dist2_squared<I: Instance<Self>>(&self, y: I) -> F;

    /// Calculate the 2-distance $\\|x-y\\|_2$, where `self` is $x$.
    #[inline]
    fn dist2<I: Instance<Self>>(&self, y: I) -> F {
        self.dist2_squared(y).sqrt()
    }

    /// Projection to the 2-ball.
    #[inline]
    fn proj_ball2(self, ρ: F) -> Self::Owned {
        let r = self.norm2();
        if r > ρ {
            self * (ρ / r)
        } else {
            self.into_owned()
        }
    }
}

pub trait ClosedEuclidean<F: Float = f64>:
    Instance<Self> + Euclidean<F, OwnedEuclidean = Self>
{
}
impl<F: Float, X: Instance<X> + Euclidean<F, OwnedEuclidean = Self>> ClosedEuclidean<F> for X {}

// TODO: remove F parameter, use AXPY::Field
pub trait EuclideanMut<F: Float = f64>: Euclidean<F> + AXPY<Field = F> {
    /// In-place projection to the 2-ball.
    #[inline]
    fn proj_ball2_mut(&mut self, ρ: F) {
        let r = self.norm2();
        if r > ρ {
            *self *= ρ / r
        }
    }
}

impl<X, F: Float> EuclideanMut<F> for X where X: Euclidean<F> + AXPY<Field = F> {}

/// Trait for [`Euclidean`] spaces with dimensions known at compile time.
pub trait StaticEuclidean<F: Float = f64>: Euclidean<F> {
    /// Returns the origin
    fn origin() -> <Self as VectorSpace>::Owned;
}

mercurial