Rust基础-错误处理-自定义错误之一

先说最简单的Rust自定义错误。本文基于Rust1.59。

实现一个最基本的自定义错误只需要实现下面两个trait,这两个都是关于把错误信息输出的。

  • 手动实现impl std::fmt::Debug的trait,一般直接添加注解即可:#[derive(Debug)]
  • 手动实现impl std::fmt::Display的trait,,用于自定义输出错误文本信息。

Talk is cheap show the code:

use std::fmt;

#[derive(Debug)]
struct AppError {
    kind: String,
    message: String,
}

impl fmt::Display for AppError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match &self.kind as &str {
            "404" => write!(f,"程序出错:{{错误类型: {}, 错误原因: {}}}",self.kind, self.message),
            _ => write!(f,"Sorry, something is wrong! Please Try Again!"),
        }
    }
}

fn produce_error() -> Result<(), AppError> {
    Err(AppError {
        kind: String::from("404"),
        message: String::from("Page not found"),
    })
}

fn main()
 {
    match produce_error(){
       Err(err) => println!("{}",err),
       _ => println!("No error"),
  }
  Ok(())
}

输出 :

 解释:

先定义一个自定义错误类型的Struct,然后为这个Struct实现Display Trait,就是怎么输出的问题。

代码没什么好说的。但问题来了

问题1: println!("{}",err),但为什么不是{:?} 或 {:#?}那种(pretty-print)把err结构打印出来?

解决方法1:导入Debug

#[derive(Debug)]
struct AppError {
    kind: String,
    message: String,
}

输出:

还是不满意?

解决方法2:把#[derive(Debug)]去掉,换自己的

impl fmt::Debug for AppError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(
            f,
            "AppError {{ 错误类型1: {}, 错误原因1: {} }}",
            self.kind, self.message
        )
    }
}

输出:

完美。

解决问题1的完整代码如下

use std::fmt;
//#[derive(Debug)]   //要使用默认Debug trait则把自定义的部分去掉
struct AppError {
    kind: String,
    message: String,
}

impl fmt::Display for AppError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match &self.kind as &str {
            "404" => write!(f,"程序出错:{{错误类型: {}, 错误原因: {}}}",self.kind, self.message),
            _ => write!(f,"Sorry, something is wrong! Please Try Again!"),
        }
    }
}
impl fmt::Debug for AppError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(
            f,
            "AppError {{ 错误类型1: {}, 错误原因1: {} }}",
            self.kind, self.message
        )
    }
}
fn produce_error() -> Result<(), AppError> {
    Err(AppError {
        kind: String::from("404"),
        message: String::from("Page not found"),
    })
}
fn main() {
    match produce_error(){
        Err(err) => println!("{:#?}",err),
        _ => println!("No error"),
    }
}

 但又有问题了,

问题2:如果类似IO,Parse之类的系统错误怎么办?

先上代码:

use std::fs::File;
use std::io::{self, Read};
use std::{num, error};
use std::fmt;
#[derive(Debug)]
struct AppError {
    kind: String,
    message: String,
}

impl fmt::Display for AppError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        match &self.kind as &str {
            "404" => write!(
                f,
                "程序出错:{{错误类型: {}, 错误原因: {}}}",
                self.kind, self.message
            ),
            _ => write!(f, "Sorry, something is wrong! Please Try Again!"),
        }
    }
}
//为AppError实现From::from<io::Error>的trait,这样AppError就能接住io::Error类型的error
impl From<io::Error> for AppError {
    fn from(error: io::Error) -> Self {
        AppError {
            kind: String::from("io_error"),
            message: error.to_string(),
        }
    }
}
//为AppError实现From::from<num::ParseIntError>的trait
impl From<num::ParseIntError> for AppError {
    fn from(error: num::ParseIntError) -> Self {
        AppError {
            kind: String::from("parse_error"),
            message: error.to_string(),
        }
    }
}
fn produce_error() -> Result<(), AppError> {
    Err(AppError {
        kind: String::from("404"),
        message: String::from("Page not found"),
    })
}

fn main() {
    match produce_error(){
        Err(err) => println!("{:#?}",err),
        _ => println!("No error"),
    }
     let mut content = String::new();
    match File::open("aaa.txt") {
        Err(err) => {
            let er:AppError= std::convert::From::from(err);//调用From<io::Error>
            println!("{:#?}", er);
        }
        Ok(mut e) => {
            match e.read_to_string(&mut content) {
                Err(err) => println!("{:#?}", err),
                Ok(_) => {
                    let number: usize = match content.parse() {
                        Ok(num) => {
                            println!("double num is:{}",num*2);
                            num
                        },
                        Err(err) => {
                            let er:AppError= std::convert::From::from(err);//调用From<num::ParseIntError>
                            println!("{:#?}", er);
                            0
                        }
                    };
                },
            };
        }
    };
}

为了让AppError"接住"io::Error和num::ParseIntError这样的系统错误,需要为AppError实现From<T>,那么当程序遇到io::Error时,通过match->Err(e)接住这个错误。所以定义了两个Trait

1、impl From<io::Error> for AppError

2、impl From<num::ParseIntError> for AppError

另外要实现上面match的操作输出自定义的AppError,就要让io::Error转换为AppError,否则就会输出默认的Error信息,那么就要 let er:AppError= std::convert::From::from(err)这样转换,这样就把AppError给“挂”上去了。

试一下:当把文件名改为abc.txt这个不存在的文件后,结果如下:

但存在2个问题

1、match的嵌套太销魂了。

2、error的返回每次都要转换,无法标准化。

下一篇介绍如何标准化自定义错误。

相关文章:

Rust基础-错误处理-自定义错误之二_DarcyZ_SSM的博客-CSDN博客

Rust基础-错误处理-自定义错误之三-snafu方案_DarcyZ_SSM的博客-CSDN博客 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Rust 中,你可以使用 `actix-web` 框架提供的 `Responder` trait 来统一处理错误响应并返回自定义的 JSON 正文。通过实现 `Responder` trait,你可以定义一个自定义的响应类型,并在处理错误时返回该响应类型的实例。 以下是一个示例,展示如何使用 `Responder` trait 在处理错误时返回自定义的 JSON 正文: ```rust use actix_web::{HttpResponse, Responder, http::StatusCode}; use serde::Serialize; #[derive(Serialize)] struct ErrorResponse { error: String, } impl Responder for ErrorResponse { type Error = actix_web::Error; type Future = futures::future::Ready<Result<HttpResponse, actix_web::Error>>; fn respond_to(self, _: &actix_web::HttpRequest) -> Self::Future { let json_body = serde_json::to_string(&self) .map_err(actix_web::error::ErrorInternalServerError) .map(|json| { HttpResponse::build(StatusCode::BAD_REQUEST) .content_type("application/json") .body(json) }); futures::future::ready(json_body) } } fn handle_request() -> impl Responder { // 在处理错误时返回自定义的 JSON 正文 let error_response = ErrorResponse { error: "Invalid request".to_string() }; error_response } ``` 在上面的示例中,我们首先定义了一个名为 `ErrorResponse` 的结构体,并使用 `serde` 的 `Serialize` trait 来自动实现 JSON 序列化。然后,我们为 `ErrorResponse` 实现了 `Responder` trait,其中的 `respond_to` 方法用于构建自定义的 JSON 响应。 在 `handle_request` 函数中,我们创建了一个 `ErrorResponse` 实例,并直接返回它作为 `Responder`。当发生错误时,`actix-web` 框架将自动调用 `respond_to` 方法来处理错误并返回经过定制的 JSON 响应。 通过这种方式,你可以在处理错误时,统一使用自定义的 JSON 正文作为响应体,并且可以根据需要定制错误的状态码、头部等其他属性。 希望这能帮到你!如果你有更多关于 `actix-web` 或 `Responder` trait 的问题,请继续提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值