#![doc(include = "../../doc/references.md")]
use crate::magma::*;
use crate::numeric::*;
pub trait AddSemigroup: AddMagma {}
pub trait AddSemigroupLaws: AddSemigroup {
fn associativity(&self, x: &Self, y: &Self) -> bool {
self.add(&x.add(y)) == self.add(x).add(y)
}
}
impl<S: AddSemigroup> AddSemigroupLaws for S {}
pub trait NumAddSemigroupLaws: NumEq + AddSemigroup {
fn num_associativity(&self, x: &Self, y: &Self, eps: &Self::Eps) -> bool {
self.add(&x.add(y)).num_eq(&self.add(x).add(y), eps)
}
}
impl<S: NumEq + AddSemigroup> NumAddSemigroupLaws for S {}
macro_rules! numeric_add_semigroup {
($type:ty) => {
impl AddSemigroup for $type {}
};
($type:ty, $($others:ty),+) => {
numeric_add_semigroup! {$type}
numeric_add_semigroup! {$($others),+}
};
}
numeric_add_semigroup! {
u8, u16, u32, u64, u128, usize, i8, i16, i32, i64, i128, isize, f32, f64
}
impl AddSemigroup for () {}
impl<A: AddSemigroup> AddSemigroup for (A,) {}
impl<A: AddSemigroup, B: AddSemigroup> AddSemigroup for (A, B) {}
impl<A: AddSemigroup, B: AddSemigroup, C: AddSemigroup> AddSemigroup for (A, B, C) {}
macro_rules! array_add_semigroup {
($size:expr) => {
impl<T: Copy + AddSemigroup> AddSemigroup for [T; $size] {}
};
($size:expr, $($others:expr),+) => {
array_add_semigroup! {$size}
array_add_semigroup! {$($others),+}
};
}
array_add_semigroup! {
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16
}
#[cfg(test)]
mod tests;