结果 Result

我们已经看到 Option 枚举类型可以用作可能失败的函数的返回值,其中返回 None 可以表明失败。但是有时要强调为什么一个操作会失败。为做到这点,我们提供 了 Result 枚举类型。

Result<T, E> 类型拥有两个取值:

  • Ok(value) 表示操作成功,并包装操作返回的 valuevalue 拥有 T 类型)。
  • Err(why),表示操作失败,并包装 why,它(但愿)能够解释失败的原因(why 拥有 E 类型)。
  1. mod checked {
  2. // 我们想要捕获的数学 “错误”
  3. #[derive(Debug)]
  4. pub enum MathError {
  5. DivisionByZero,
  6. NegativeLogarithm,
  7. NegativeSquareRoot,
  8. }
  9. pub type MathResult = Result<f64, MathError>;
  10. pub fn div(x: f64, y: f64) -> MathResult {
  11. if y == 0.0 {
  12. // 此操作将会失败,那么(与其让程序崩溃)不如把失败的原因包装在
  13. // `Err` 中并返回
  14. Err(MathError::DivisionByZero)
  15. } else {
  16. // 此操作是有效的,返回包装在 `Ok` 中的结果
  17. Ok(x / y)
  18. }
  19. }
  20. pub fn sqrt(x: f64) -> MathResult {
  21. if x < 0.0 {
  22. Err(MathError::NegativeSquareRoot)
  23. } else {
  24. Ok(x.sqrt())
  25. }
  26. }
  27. pub fn ln(x: f64) -> MathResult {
  28. if x < 0.0 {
  29. Err(MathError::NegativeLogarithm)
  30. } else {
  31. Ok(x.ln())
  32. }
  33. }
  34. }
  35. // `op(x, y)` === `sqrt(ln(x / y))`
  36. fn op(x: f64, y: f64) -> f64 {
  37. // 这是一个三层的 match 金字塔!
  38. match checked::div(x, y) {
  39. Err(why) => panic!("{:?}", why),
  40. Ok(ratio) => match checked::ln(ratio) {
  41. Err(why) => panic!("{:?}", why),
  42. Ok(ln) => match checked::sqrt(ln) {
  43. Err(why) => panic!("{:?}", why),
  44. Ok(sqrt) => sqrt,
  45. },
  46. },
  47. }
  48. }
  49. fn main() {
  50. // 这会失败吗?
  51. println!("{}", op(1.0, 10.0));
  52. }