单元测试

测试(test)是这样一种 Rust 函数:它保证其他部分的代码按照所希望的行为正常 运行。测试函数的函数体通常会进行一些配置,运行我们想要测试的代码,然后 断言(assert)结果是不是我们所期望的。

大多数单元测试都会被放到一个叫 tests 的、带有 #[cfg(test)] 属性 的模块中,测试函数要加上 #[test] 属性。

当测试函数中有什么东西 panic 了,测试就失败。有一些这方面的 辅助

  • assert!(expression) - 如果表达式的值是 false 则 panic。
  • assert_eq!(left, right)assert_ne!(left, right) - 检验左右两边是否 相等/不等。
  1. pub fn add(a: i32, b: i32) -> i32 {
  2. a + b
  3. }
  4. // 这个加法函数写得很差,本例中我们会使它失败。
  5. #[allow(dead_code)]
  6. fn bad_add(a: i32, b: i32) -> i32 {
  7. a - b
  8. }
  9. #[cfg(test)]
  10. mod tests {
  11. // 注意这个惯用法:在 tests 模块中,从外部作用域导入所有名字。
  12. use super::*;
  13. #[test]
  14. fn test_add() {
  15. assert_eq!(add(1, 2), 3);
  16. }
  17. #[test]
  18. fn test_bad_add() {
  19. // 这个断言会导致测试失败。注意私有的函数也可以被测试!
  20. assert_eq!(bad_add(1, 2), 3);
  21. }
  22. }

可以使用 cargo test 来运行测试。

  1. $ cargo test
  2. running 2 tests
  3. test tests::test_bad_add ... FAILED
  4. test tests::test_add ... ok
  5. failures:
  6. ---- tests::test_bad_add stdout ----
  7. thread 'tests::test_bad_add' panicked at 'assertion failed: `(left == right)`
  8. left: `-1`,
  9. right: `3`', src/lib.rs:21:8
  10. note: Run with `RUST_BACKTRACE=1` for a backtrace.
  11. failures:
  12. tests::test_bad_add
  13. test result: FAILED. 1 passed; 1 failed; 0 ignored; 0 measured; 0 filtered out

测试 panic

一些函数应当在特定条件下 panic。为测试这种行为,请使用 #[should_panic] 属性。这 个属性接受可选参数 expected = 以指定 panic 时的消息。如果你的函数能以多种方式 panic,这个属性就保证了你在测试的确实是所指定的 panic。

  1. pub fn divide_non_zero_result(a: u32, b: u32) -> u32 {
  2. if b == 0 {
  3. panic!("Divide-by-zero error");
  4. } else if a < b {
  5. panic!("Divide result is zero");
  6. }
  7. a / b
  8. }
  9. #[cfg(test)]
  10. mod tests {
  11. use super::*;
  12. #[test]
  13. fn test_divide() {
  14. assert_eq!(divide_non_zero_result(10, 2), 5);
  15. }
  16. #[test]
  17. #[should_panic]
  18. fn test_any_panic() {
  19. divide_non_zero_result(1, 0);
  20. }
  21. #[test]
  22. #[should_panic(expected = "Divide result is zero")]
  23. fn test_specific_panic() {
  24. divide_non_zero_result(1, 10);
  25. }
  26. }

运行这些测试会输出:

  1. $ cargo test
  2. running 3 tests
  3. test tests::test_any_panic ... ok
  4. test tests::test_divide ... ok
  5. test tests::test_specific_panic ... ok
  6. test result: ok. 3 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out
  7. Doc-tests tmp-test-should-panic
  8. running 0 tests
  9. test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out

运行特定的测试

要运行特定的测试,只要把测试名称传给 cargo test 命令就可以了。

  1. $ cargo test test_any_panic
  2. running 1 test
  3. test tests::test_any_panic ... ok
  4. test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 2 filtered out
  5. Doc-tests tmp-test-should-panic
  6. running 0 tests
  7. test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out

要运行多个测试,可以仅指定测试名称中的一部分,用它来匹配所有要运行的测试。

  1. $ cargo test panic
  2. running 2 tests
  3. test tests::test_any_panic ... ok
  4. test tests::test_specific_panic ... ok
  5. test result: ok. 2 passed; 0 failed; 0 ignored; 0 measured; 1 filtered out
  6. Doc-tests tmp-test-should-panic
  7. running 0 tests
  8. test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out

忽略测试

可以把属性 #[ignore] 赋予测试以排除某些测试,或者使用 cargo test -- --ignored 命令来运行它们。

  1. pub fn add(a: i32, b: i32) -> i32 {
  2. a + b
  3. }
  4. #[cfg(test)]
  5. mod tests {
  6. use super::*;
  7. #[test]
  8. fn test_add() {
  9. assert_eq!(add(2, 2), 4);
  10. }
  11. #[test]
  12. fn test_add_hundred() {
  13. assert_eq!(add(100, 2), 102);
  14. assert_eq!(add(2, 100), 102);
  15. }
  16. #[test]
  17. #[ignore]
  18. fn ignored_test() {
  19. assert_eq!(add(0, 0), 0);
  20. }
  21. }
  1. $ cargo test
  2. running 1 test
  3. test tests::ignored_test ... ignored
  4. test result: ok. 0 passed; 0 failed; 1 ignored; 0 measured; 0 filtered out
  5. Doc-tests tmp-ignore
  6. running 0 tests
  7. test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out
  8. $ cargo test -- --ignored
  9. running 1 test
  10. test tests::ignored_test ... ok
  11. test result: ok. 1 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out
  12. Doc-tests tmp-ignore
  13. running 0 tests
  14. test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out