Result

Optionenumで失敗する可能性のある関数から値を返すことができました。 その場合、失敗したときはNoneを返します。しかし、しばしばなぜ関数が 失敗したのかが重要になってくることがあります。これはResultenumで表現します。

Result<T, E>enumは2つの列挙子を持ちます。

  • Ok(value)は成功時に返され、T型の値valueには結果が入ります。
  • Err(why)は失敗時に返され、E型の値whyには失敗の原因が入ります。
mod checked {
    // 数学的な「エラー」をキャッチする
    #[derive(Debug)]
    pub enum MathError {
        DivisionByZero,
        NonPositiveLogarithm,
        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::NonPositiveLogarithm)
        } else {
            Ok(x.ln())
        }
    }
}

// `op(x, y)`と`sqrt(ln(x / y))`は等価です
fn op(x: f64, y: f64) -> f64 {
    // 3段階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));
}