Rust错误处理

简单的错误处理

要不是使用“?”,Rust的错误处理会显得有些不够灵巧。要使用“?”我们需要声明返回值类型为Result类型,这种类型可以包含任何具备std::error::Error特征从而可以转换为Bax类型的错误类型。
拿我们需要处理IO错误和字符串转换为数字错误举例:

use std::fs::File;
use std::io::prelude::*;
use std::error::Error;

fn run(file: &str) -> Result> {
   let mut file = File::open(file)?;
   let mut contents = String::new();
   file.read_to_string(&mut contents)?;
   Ok(contents.trim().parse()?)
}

这里使用了俩个"?"处理可能发生的IO错误:打开文件错误和读取内容为string错误。
使用了一个“?”处理可能发生的类型转换错误。
最后我们将结果包装为Ok类型。Rust可以中返回值中判断出parse的结果为i32类型。
简化Result类型的声明比较容易,我们可以定义一个自己的Result类型,比如:

type BoxResult = Result>

但是,我们程序还需要自定义Error类型,那我们就需要做些其他工作:

  • 可以实现Debug特征
  • 必须实现Display特征
  • 必须实现Error特征

就像这样:

//error1.rs
use std::error::Error;
use std::fmt;

#[derive(Debug)]
struct MyError {
   details: String
}

impl MyError {
        fn new(msg: &str) -> MyError {
              MyError{details: msg.to_string()}
        }
}
impl fnt::Display for MyError {
         fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Reuslt {
              write!(f, "{}", self.details)
      }
}

impls Error for MyError {
       fn description(&self)  -> &str {
                  &self.details
       }

}

//一个测试我们自定义错误的函数
fn raises_my_error(yes: bool) -> Result<(), MyError> {
       if yes {
              Err(MyError::new("borked"))
       } else {
                Ok(())
       }
}

输入Result比较麻烦,所以很多模块都定义了自己的Result类型,这样可以少敲几下键盘。比如IO模块定义了io::Reust代替Result使用。
在下面的例子中演示的是怎么处理String类型转化为浮点数类型可能出现错误的情况。
现在我们知道使用"?"可以方便的把有错误的表达式转换为Err返回。这种转换是通过From特征实现的。
你可以继续使用这种方便的转换,这在一些比较简单的应用中是个不错的选择,接下来我们演示的场景会复杂一些。
ParseFloatError实现了Error特征,所以它具备description()方法。

use std::num::ParseFloatError;
impl From for MyError {
       fn from(err: ParseFloatError) -> Self {
                 MyError::new(err.description())
       }
}

fn parse_f64(s: &str,  yes: bool) -> Result {
         raise_my_error(yes)?;
         let x: f64 = s.parse()?
         Ok(x)
}

fn main() {
    println!(" {:?}", parse_f64("42", false));
    println!(" {:?}", parse_f64("42", true));
    println!(" {:?}", parse_64("?42", false));
}

执行结果会是这样:

Ok(42)
Err(MyError {details: “borked”})
Err(MyError {details: “invalid float literal”})

未完待续

原文地址

你可能感兴趣的:(学习新时代编程语言Rust)