在Rust中编写自定义Error的详细代码
作者:归云十九少
Result<T, E> 类型可以方便地用于错误传导,Result<T, E>是模板类型,实例化后可以是各种类型,但 Rust 要求传导的 Result 中的 E 是相同类型的,所以我们需要编写自己的 Error 类型,本文给大家介绍了在Rust中编写自定义Error的详细代码,需要的朋友可以参考下
前言
之前我们聊过,Result<T, E> 类型可以方便地用于错误传导,Result<T, E>是模板类型,实例化后可以是各种类型,但 Rust 要求传导的 Result 中的 E 是相同类型的,或者能够自动转化为相同类型。比如,下面这段代码编译就会报错。所以我们需要编写自己的 Error 类型,以同时包含系统错误和具体业务错误。
use std::io; use std::fs::{File}; fn read_fs() -> io::Result<()> { File::open("abc.txt")?; Ok(()) } fn user_err() -> Result<u32, String> { Err(String::from("test faill")) } fn test() -> Result<u32, String>{ read_fs()?; test()?; } pub fn main() { test(); }
系统Error
在 Result<T, E> 中,E 表示一种错误,Rust 标准库已经定义一系列 Error,主要是 io Error。它的定义如下:
pub struct Error { repr: Repr, }
我们可以通过它的关联函数 new,from,other,配合 ErrorKind 来生成一个io Error 用于Reuslt。
自定义 Error
我们要定义一个自己的 Error 类型,它既包含系统的 io Error,也包含业务 Error,定义如下
// 业务 Error 定义 enum WorkError { WorkErrorFirst, WorkErrorSecond, WorkErrorThird, } // 复合 Error 定义 enum MyError { IoErr(std::io::Error), WorkErr(WorkError) }
使用 MyError
fn read_fs() -> Result<(), MyError> { let ret = File::open("abc.txt"); if let Err(e) = ret { return Err(MyError::IoErr(e)); } else { println!("ret = {ret:?}"); } Ok(()) } fn user_err() -> Result<(), MyError> { Err(MyError::WorkErr(WorkError::WorkErrorSecond)) } fn test() -> Result<(), MyError>{ read_fs()?; user_err()?; Ok(()) } pub fn main() { let ret = test(); match ret { Ok(_) => { println!("run test success"); } Err(e) => { println!("run test fail"); } } }
在上述实现中,我们通过手动方式将 std::io::Error 转化为 MyError。也可以通过实现 From trait 来自动实现这种转化。
impl From<std::io::Error> for MyError { fn from(err: std::io::Error) -> Self { MyError::IoErr(err) } } // 这样,read_fs 函数就可以简化如下: fn read_fs() -> Result<(), MyError> { File::open("abc.txt")?; Ok(()) }
现在还有一个问题,在 main 函数中,我们无法打印 e 信息。通过简单的为 WorkError 和 MyError 类型增加 #[derive(Debug)] 声明,就可以直接打印 e 信息。
到此这篇关于在Rust中编写自定义Error的详细代码的文章就介绍到这了,更多相关Rust编写自定义Error内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!