From 3ec194bafbae7d479effee44074d4ab5ee98e42c Mon Sep 17 00:00:00 2001 From: bluss Date: Fri, 15 Apr 2016 13:15:28 +0200 Subject: [PATCH 1/4] complex: implement OpAssign for Complex --- complex/src/lib.rs | 82 ++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 82 insertions(+) diff --git a/complex/src/lib.rs b/complex/src/lib.rs index cb605a2..9ab4ef9 100644 --- a/complex/src/lib.rs +++ b/complex/src/lib.rs @@ -516,6 +516,88 @@ impl Div> for Complex { } } +// Op Assign + +mod opassign { + use std::ops::{AddAssign, SubAssign, MulAssign, DivAssign}; + + use traits::Num; + + use Complex; + + impl AddAssign for Complex { + fn add_assign(&mut self, other: Complex) { + *self = self.clone() + other; + } + } + + impl SubAssign for Complex { + fn sub_assign(&mut self, other: Complex) { + *self = self.clone() - other; + } + } + + impl MulAssign for Complex { + fn mul_assign(&mut self, other: Complex) { + *self = self.clone() * other; + } + } + + impl DivAssign for Complex { + fn div_assign(&mut self, other: Complex) { + *self = self.clone() / other; + } + } + + impl AddAssign for Complex { + fn add_assign(&mut self, other: T) { + self.re += other; + } + } + + impl SubAssign for Complex { + fn sub_assign(&mut self, other: T) { + self.re -= other; + } + } + + impl MulAssign for Complex { + fn mul_assign(&mut self, other: T) { + self.re *= other.clone(); + self.im *= other; + } + } + + impl DivAssign for Complex { + fn div_assign(&mut self, other: T) { + self.re /= other.clone(); + self.im /= other; + } + } + + macro_rules! forward_op_assign { + (impl $imp:ident, $method:ident) => { + impl<'a, T: Clone + Num> $imp<&'a Complex> for Complex { + #[inline] + fn $method(&mut self, other: &Complex) { + self.$method(other.clone()) + } + } + impl<'a, T: Clone + Num + $imp> $imp<&'a T> for Complex { + #[inline] + fn $method(&mut self, other: &T) { + self.$method(other.clone()) + } + } + } + } + + forward_op_assign!(impl AddAssign, add_assign); + forward_op_assign!(impl SubAssign, sub_assign); + forward_op_assign!(impl MulAssign, mul_assign); + forward_op_assign!(impl DivAssign, div_assign); +} + impl> Neg for Complex { type Output = Complex; From 686411f44e7e589ec2b1fe733c2de27f26a4e47c Mon Sep 17 00:00:00 2001 From: bluss Date: Fri, 15 Apr 2016 13:15:28 +0200 Subject: [PATCH 2/4] complex: Add tests for OpAssign Add tests by expanding the current a + b tests to include a += b, and so on. --- complex/src/lib.rs | 80 +++++++++++++++++++++++++++++++--------------- 1 file changed, 55 insertions(+), 25 deletions(-) diff --git a/complex/src/lib.rs b/complex/src/lib.rs index 9ab4ef9..e43525e 100644 --- a/complex/src/lib.rs +++ b/complex/src/lib.rs @@ -870,6 +870,7 @@ mod test { pub const _neg1_1i : Complex64 = Complex { re: -1.0, im: 1.0 }; pub const _05_05i : Complex64 = Complex { re: 0.5, im: 0.5 }; pub const all_consts : [Complex64; 5] = [_0_0i, _1_0i, _1_1i, _neg1_1i, _05_05i]; + pub const _4_2i : Complex64 = Complex { re: 4.0, im: 2.0 }; #[test] fn test_consts() { @@ -1299,55 +1300,83 @@ mod test { } } + // Test both a + b and a += b + macro_rules! test_a_op_b { + ($a:ident + $b:expr, $answer:expr) => { + assert_eq!($a + $b, $answer); + assert_eq!({ let mut x = $a; x += $b; x}, $answer); + }; + ($a:ident - $b:expr, $answer:expr) => { + assert_eq!($a - $b, $answer); + assert_eq!({ let mut x = $a; x -= $b; x}, $answer); + }; + ($a:ident * $b:expr, $answer:expr) => { + assert_eq!($a * $b, $answer); + assert_eq!({ let mut x = $a; x *= $b; x}, $answer); + }; + ($a:ident / $b:expr, $answer:expr) => { + assert_eq!($a / $b, $answer); + assert_eq!({ let mut x = $a; x /= $b; x}, $answer); + }; + } + + // Test both a + b and a + &b + macro_rules! test_op { + ($a:ident $op:tt $b:expr, $answer:expr) => { + test_a_op_b!($a $op $b, $answer); + test_a_op_b!($a $op &$b, $answer); + } + } + mod complex_arithmetic { use super::{_0_0i, _1_0i, _1_1i, _0_1i, _neg1_1i, _05_05i, all_consts}; use traits::Zero; #[test] fn test_add() { - assert_eq!(_05_05i + _05_05i, _1_1i); - assert_eq!(_0_1i + _1_0i, _1_1i); - assert_eq!(_1_0i + _neg1_1i, _0_1i); + test_op!(_05_05i + _05_05i, _1_1i); + test_op!(_0_1i + _1_0i, _1_1i); + test_op!(_1_0i + _neg1_1i, _0_1i); for &c in all_consts.iter() { - assert_eq!(_0_0i + c, c); - assert_eq!(c + _0_0i, c); + test_op!(_0_0i + c, c); + test_op!(c + _0_0i, c); } } #[test] fn test_sub() { - assert_eq!(_05_05i - _05_05i, _0_0i); - assert_eq!(_0_1i - _1_0i, _neg1_1i); - assert_eq!(_0_1i - _neg1_1i, _1_0i); + test_op!(_05_05i - _05_05i, _0_0i); + test_op!(_0_1i - _1_0i, _neg1_1i); + test_op!(_0_1i - _neg1_1i, _1_0i); for &c in all_consts.iter() { - assert_eq!(c - _0_0i, c); - assert_eq!(c - c, _0_0i); + test_op!(c - _0_0i, c); + test_op!(c - c, _0_0i); } } #[test] fn test_mul() { - assert_eq!(_05_05i * _05_05i, _0_1i.unscale(2.0)); - assert_eq!(_1_1i * _0_1i, _neg1_1i); + test_op!(_05_05i * _05_05i, _0_1i.unscale(2.0)); + test_op!(_1_1i * _0_1i, _neg1_1i); // i^2 & i^4 - assert_eq!(_0_1i * _0_1i, -_1_0i); + test_op!(_0_1i * _0_1i, -_1_0i); assert_eq!(_0_1i * _0_1i * _0_1i * _0_1i, _1_0i); for &c in all_consts.iter() { - assert_eq!(c * _1_0i, c); - assert_eq!(_1_0i * c, c); + test_op!(c * _1_0i, c); + test_op!(_1_0i * c, c); } } #[test] fn test_div() { - assert_eq!(_neg1_1i / _0_1i, _1_1i); + test_op!(_neg1_1i / _0_1i, _1_1i); for &c in all_consts.iter() { if c != Zero::zero() { - assert_eq!(c / c, _1_0i); + test_op!(c / c, _1_0i); } } } @@ -1364,29 +1393,30 @@ mod test { mod real_arithmetic { use super::super::Complex; + use super::_4_2i; #[test] fn test_add() { - assert_eq!(Complex::new(4.0, 2.0) + 0.5, Complex::new(4.5, 2.0)); - assert_eq!(0.5 + Complex::new(4.0, 2.0), Complex::new(4.5, 2.0)); + test_op!(_4_2i + 0.5, Complex::new(4.5, 2.0)); + assert_eq!(0.5 + _4_2i, Complex::new(4.5, 2.0)); } #[test] fn test_sub() { - assert_eq!(Complex::new(4.0, 2.0) - 0.5, Complex::new(3.5, 2.0)); - assert_eq!(0.5 - Complex::new(4.0, 2.0), Complex::new(-3.5, -2.0)); + test_op!(_4_2i - 0.5, Complex::new(3.5, 2.0)); + assert_eq!(0.5 - _4_2i, Complex::new(-3.5, -2.0)); } #[test] fn test_mul() { - assert_eq!(Complex::new(4.0, 2.0) * 0.5, Complex::new(2.0, 1.0)); - assert_eq!(0.5 * Complex::new(4.0, 2.0), Complex::new(2.0, 1.0)); + assert_eq!(_4_2i * 0.5, Complex::new(2.0, 1.0)); + assert_eq!(0.5 * _4_2i, Complex::new(2.0, 1.0)); } #[test] fn test_div() { - assert_eq!(Complex::new(4.0, 2.0) / 0.5, Complex::new(8.0, 4.0)); - assert_eq!(0.5 / Complex::new(4.0, 2.0), Complex::new(0.1, -0.05)); + assert_eq!(_4_2i / 0.5, Complex::new(8.0, 4.0)); + assert_eq!(0.5 / _4_2i, Complex::new(0.1, -0.05)); } } From 055c6693b15954ded07878065c4a6e2a43bb27c6 Mon Sep 17 00:00:00 2001 From: Josh Stone Date: Wed, 12 Jul 2017 13:54:57 -0700 Subject: [PATCH 3/4] complex: require T:NumAssign for assign ops --- complex/src/lib.rs | 22 +++++++++++----------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/complex/src/lib.rs b/complex/src/lib.rs index e43525e..b0c2e4c 100644 --- a/complex/src/lib.rs +++ b/complex/src/lib.rs @@ -521,54 +521,54 @@ impl Div> for Complex { mod opassign { use std::ops::{AddAssign, SubAssign, MulAssign, DivAssign}; - use traits::Num; + use traits::NumAssign; use Complex; - impl AddAssign for Complex { + impl AddAssign for Complex { fn add_assign(&mut self, other: Complex) { *self = self.clone() + other; } } - impl SubAssign for Complex { + impl SubAssign for Complex { fn sub_assign(&mut self, other: Complex) { *self = self.clone() - other; } } - impl MulAssign for Complex { + impl MulAssign for Complex { fn mul_assign(&mut self, other: Complex) { *self = self.clone() * other; } } - impl DivAssign for Complex { + impl DivAssign for Complex { fn div_assign(&mut self, other: Complex) { *self = self.clone() / other; } } - impl AddAssign for Complex { + impl AddAssign for Complex { fn add_assign(&mut self, other: T) { self.re += other; } } - impl SubAssign for Complex { + impl SubAssign for Complex { fn sub_assign(&mut self, other: T) { self.re -= other; } } - impl MulAssign for Complex { + impl MulAssign for Complex { fn mul_assign(&mut self, other: T) { self.re *= other.clone(); self.im *= other; } } - impl DivAssign for Complex { + impl DivAssign for Complex { fn div_assign(&mut self, other: T) { self.re /= other.clone(); self.im /= other; @@ -577,13 +577,13 @@ mod opassign { macro_rules! forward_op_assign { (impl $imp:ident, $method:ident) => { - impl<'a, T: Clone + Num> $imp<&'a Complex> for Complex { + impl<'a, T: Clone + NumAssign> $imp<&'a Complex> for Complex { #[inline] fn $method(&mut self, other: &Complex) { self.$method(other.clone()) } } - impl<'a, T: Clone + Num + $imp> $imp<&'a T> for Complex { + impl<'a, T: Clone + NumAssign> $imp<&'a T> for Complex { #[inline] fn $method(&mut self, other: &T) { self.$method(other.clone()) From d57c0c2879cc1bb6e48dda636eac3dc65d1695d2 Mon Sep 17 00:00:00 2001 From: Josh Stone Date: Wed, 12 Jul 2017 13:59:30 -0700 Subject: [PATCH 4/4] complex: implement add/sub_assign directly --- complex/src/lib.rs | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/complex/src/lib.rs b/complex/src/lib.rs index b0c2e4c..a47a7e1 100644 --- a/complex/src/lib.rs +++ b/complex/src/lib.rs @@ -527,13 +527,15 @@ mod opassign { impl AddAssign for Complex { fn add_assign(&mut self, other: Complex) { - *self = self.clone() + other; + self.re += other.re; + self.im += other.im; } } impl SubAssign for Complex { fn sub_assign(&mut self, other: Complex) { - *self = self.clone() - other; + self.re -= other.re; + self.im -= other.im; } }