DRY (不写重复代码)

通过提取函数或测试集的公共部分,宏可以让你写出 DRY 的代码(DRY 是 Don’t Repeat Yourself 的缩写,意思为 “不要写重复代码”)。这里给出一个例子,对 Vec<T> 实现 并测试了关于 +=*=-= 等运算符。

  1. use std::ops::{Add, Mul, Sub};
  2. macro_rules! assert_equal_len {
  3. // `tt`(token tree,标记树)指示符表示运算符和标记。
  4. ($a:ident, $b: ident, $func:ident, $op:tt) => (
  5. assert!($a.len() == $b.len(),
  6. "{:?}: dimension mismatch: {:?} {:?} {:?}",
  7. stringify!($func),
  8. ($a.len(),),
  9. stringify!($op),
  10. ($b.len(),));
  11. )
  12. }
  13. macro_rules! op {
  14. ($func:ident, $bound:ident, $op:tt, $method:ident) => (
  15. fn $func<T: $bound<T, Output=T> + Copy>(xs: &mut Vec<T>, ys: &Vec<T>) {
  16. assert_equal_len!(xs, ys, $func, $op);
  17. for (x, y) in xs.iter_mut().zip(ys.iter()) {
  18. *x = $bound::$method(*x, *y);
  19. // *x = x.$method(*y);
  20. }
  21. }
  22. )
  23. }
  24. // 实现 `add_assign`、`mul_assign` 和 `sub_assign` 等函数。
  25. op!(add_assign, Add, +=, add);
  26. op!(mul_assign, Mul, *=, mul);
  27. op!(sub_assign, Sub, -=, sub);
  28. mod test {
  29. use std::iter;
  30. macro_rules! test {
  31. ($func: ident, $x:expr, $y:expr, $z:expr) => {
  32. #[test]
  33. fn $func() {
  34. for size in 0usize..10 {
  35. let mut x: Vec<_> = iter::repeat($x).take(size).collect();
  36. let y: Vec<_> = iter::repeat($y).take(size).collect();
  37. let z: Vec<_> = iter::repeat($z).take(size).collect();
  38. super::$func(&mut x, &y);
  39. assert_eq!(x, z);
  40. }
  41. }
  42. }
  43. }
  44. // 测试 `add_assign`、`mul_assign` 和 `sub_assign`
  45. test!(add_assign, 1u32, 2u32, 3u32);
  46. test!(mul_assign, 2u32, 3u32, 6u32);
  47. test!(sub_assign, 3u32, 2u32, 1u32);
  48. }
  1. $ rustc --test dry.rs && ./dry
  2. running 3 tests
  3. test test::mul_assign ... ok
  4. test test::add_assign ... ok
  5. test test::sub_assign ... ok
  6. test result: ok. 3 passed; 0 failed; 0 ignored; 0 measured