结果 Result
我们已经看到 Option 枚举类型可以用作可能失败的函数的返回值,其中返回 None
可以表明失败。但是有时要强调为什么一个操作会失败。为做到这点,我们提供
了 Result 枚举类型。
Result<T, E> 类型拥有两个取值:
Ok(value)表示操作成功,并包装操作返回的value(value拥有T类型)。Err(why),表示操作失败,并包装why,它(但愿)能够解释失败的原因(why拥有E类型)。
mod checked {// 我们想要捕获的数学 “错误”#[derive(Debug)]pub enum MathError {DivisionByZero,NegativeLogarithm,NegativeSquareRoot,}pub type MathResult = Result<f64, MathError>;pub fn div(x: f64, y: f64) -> MathResult {if y == 0.0 {// 此操作将会失败,那么(与其让程序崩溃)不如把失败的原因包装在// `Err` 中并返回Err(MathError::DivisionByZero)} else {// 此操作是有效的,返回包装在 `Ok` 中的结果Ok(x / y)}}pub fn sqrt(x: f64) -> MathResult {if x < 0.0 {Err(MathError::NegativeSquareRoot)} else {Ok(x.sqrt())}}pub fn ln(x: f64) -> MathResult {if x < 0.0 {Err(MathError::NegativeLogarithm)} else {Ok(x.ln())}}}// `op(x, y)` === `sqrt(ln(x / y))`fn op(x: f64, y: f64) -> f64 {// 这是一个三层的 match 金字塔!match checked::div(x, y) {Err(why) => panic!("{:?}", why),Ok(ratio) => match checked::ln(ratio) {Err(why) => panic!("{:?}", why),Ok(ln) => match checked::sqrt(ln) {Err(why) => panic!("{:?}", why),Ok(sqrt) => sqrt,},},}}fn main() {// 这会失败吗?println!("{}", op(1.0, 10.0));}
