Skip to content

Commit 06c0ee0

Browse files
tarciericuviper
authored andcommitted
Add ZeroConstant and OneConstant traits
Adds traits which are alternatives to the more dynamic `Zero`/`One` traits which are useful for stack-allocated types where it's possible to define constant values for zero/one. `ZeroConstant` bounds on `Zero` as a supertrait, and `OneConstant` on `One`, allowing them to be used as drop-in replacements. When a type also impls `PartialEq`, then `ZeroConstant` also provides a blanket impl of `Zero`, and likewise for `OneConstant`/`One`, making it simple for stack-allocated integers to impl these traits as an alternative to `Zero`/`One` while still remaining fully compatible. The internal impls of `Zero`/`One` on the numeric primitive types have been changed to use these traits, which should be a fully backwards-compatible change.
1 parent 29c5b46 commit 06c0ee0

File tree

1 file changed

+48
-18
lines changed

1 file changed

+48
-18
lines changed

src/identities.rs

+48-18
Original file line numberDiff line numberDiff line change
@@ -28,17 +28,32 @@ pub trait Zero: Sized + Add<Self, Output = Self> {
2828
fn is_zero(&self) -> bool;
2929
}
3030

31+
/// Defines an associated constant representing the additive identity element
32+
/// for `Self`.
33+
///
34+
/// Types which impl both this trait and [`PartialEq`] will receive a blanket
35+
/// impl of the [`Zero`] trait.
36+
pub trait ZeroConstant: Zero {
37+
/// The additive identity element of `Self`, `0`.
38+
const ZERO: Self;
39+
}
40+
41+
impl<T: ZeroConstant + PartialEq> Zero for T {
42+
#[inline(always)]
43+
fn zero() -> T {
44+
Self::ZERO
45+
}
46+
47+
#[inline(always)]
48+
fn is_zero(&self) -> bool {
49+
self == &Self::ZERO
50+
}
51+
}
52+
3153
macro_rules! zero_impl {
3254
($t:ty, $v:expr) => {
33-
impl Zero for $t {
34-
#[inline]
35-
fn zero() -> $t {
36-
$v
37-
}
38-
#[inline]
39-
fn is_zero(&self) -> bool {
40-
*self == $v
41-
}
55+
impl ZeroConstant for $t {
56+
const ZERO: Self = $v;
4257
}
4358
};
4459
}
@@ -115,17 +130,32 @@ pub trait One: Sized + Mul<Self, Output = Self> {
115130
}
116131
}
117132

133+
/// Defines an associated constant representing the multiplicative identity
134+
/// element for `Self`.
135+
///
136+
/// Types which impl both this trait and [`PartialEq`] will receive a blanket
137+
/// impl of the [`One`] trait.
138+
pub trait OneConstant: One {
139+
/// The multiplicative identity element of `Self`, `1`.
140+
const ONE: Self;
141+
}
142+
143+
impl<T: OneConstant + PartialEq> One for T {
144+
#[inline(always)]
145+
fn one() -> T {
146+
Self::ONE
147+
}
148+
149+
#[inline(always)]
150+
fn is_one(&self) -> bool {
151+
self == &Self::ONE
152+
}
153+
}
154+
118155
macro_rules! one_impl {
119156
($t:ty, $v:expr) => {
120-
impl One for $t {
121-
#[inline]
122-
fn one() -> $t {
123-
$v
124-
}
125-
#[inline]
126-
fn is_one(&self) -> bool {
127-
*self == $v
128-
}
157+
impl OneConstant for $t {
158+
const ONE: Self = $v;
129159
}
130160
};
131161
}

0 commit comments

Comments
 (0)