Rust eyre 错误处理实战教程
在《Rust 错误处理库: thiserror 和 anyhow》中我们介绍了Rust简化处理错误策略,本文解释eyre错误处理库,并通过多个实际示例进行说明,最后于anyhow库进行对比,让你更好理解其应用场景。
eyre
是一个用于 Rust 的错误处理库,它提供了方便、灵活且具有良好错误信息显示的功能。其主要目标是简化在 Rust 程序中处理错误的过程,尤其是在处理复杂的错误场景和嵌套的Result
或Option
类型时。与 Rust 标准库的错误处理相比,eyre
提供了更丰富的上下文信息。例如,当发生错误时,它可以包含更多关于错误发生位置(如文件名、行号等)的详细信息,并且在报告错误时可以提供更友好的用户界面,使得开发者更容易理解和定位错误。
读取文件示例
use std::fs::File;
use eyre::Result;
fn read_file() -> Result<String> {let mut file = File::open("example.txt")?;let mut contents = String::new();file.read_to_string(&mut contents)?;Ok(contents)
}
fn main() {match read_file() {Ok(contents) => println!("File contents: {}", contents),Err(e) => eprintln!("Error: {}", e),}
}
在read_file
函数中,首先尝试打开一个名为example.txt
的文件。File::open
函数返回一个Result
类型,?
操作符在这里用于传播错误。如果文件打开失败,?
会立即返回错误,并且eyre
会自动捕获一些关于这个错误的上下文信息,如函数调用位置等。接着,读取文件内容到一个String
变量中,同样使用?
操作符来处理read_to_string
函数可能产生的错误。
在main
函数中,通过match
语句来处理read_file
函数返回的Result
类型。如果成功,打印文件内容;如果失败,使用eprintln!
打印错误信息,eyre
提供的错误信息会包含详细的错误原因和位置相关内容。
自定义错误类型
use eyre::{Context, Result};
use std::num::ParseIntError;
#[derive(Debug)]
struct MyError {inner: ParseIntError,context: String,
}
impl std::fmt::Display for MyError {fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {write!(f, "Error in context '{}': {}", self.context, self.inner)}
}
impl std::error::Error for MyError {fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {Some(&self.inner)}
}
fn parse_number(s: &str) -> Result<i32> {let number = s.parse::<i32>().map_err(|e| MyError {inner: e,context: "Parsing number".to_string(),})?;Ok(number)
}
fn main() {match parse_number("not a number") {Ok(n) => println!("Parsed number: {}", n),Err(e) => eprintln!("Error: {}", e),}
}
首先定义了一个自定义的错误类型MyError
,它包含了一个内部错误(这里是ParseIntError
类型,用于表示数字解析错误)和一个错误上下文信息(context
字段)。实现了Display
和Error
trait,用于格式化错误信息和提供错误源信息。
在parse_number
函数中,尝试将一个字符串解析为i32
类型的数字。如果解析失败,会使用map_err
函数将ParseIntError
转换为自定义的MyError
类型,并添加错误上下文信息。同样,?
操作符用于传播错误。
在main
函数中,通过match
语句处理parse_number
函数返回的Result
类型,打印解析后的数字或者错误信息。eyre
会根据自定义的错误类型和相关实现来提供详细的错误显示,包括自定义的上下文信息和内部错误的详细内容。
异步函数错误处理
在 Rust 的异步编程中,经常会使用async
函数,这些函数返回Result
类型来表示成功或失败的结果。eyre
包可以很好地与这种异步函数结合使用,帮助我们处理错误。
use std::fs::File;
use tokio::io::AsyncReadExt;
use eyre::Result;
use tokio::task;
async fn read_file_async() -> Result<String> {let mut file = File::open("example.txt").await?;let mut contents = String::new();file.read_to_string(&mut contents).await?;Ok(contents)
}
#[tokio::main]
async fn main() {match read_file_async().await {Ok(contents) => println!("File contents: {}", contents),Err(e) => eprintln!("Error: {}", e),}
}
在read_file_async
异步函数中,首先尝试异步打开一个名为example.txt
的文件。注意这里使用了await
关键字来等待文件打开操作完成,并且?
操作符用于处理File::open
返回的Result
类型中的错误。如果文件打开失败,?
会立即返回错误,eyre
会捕获相关的错误上下文信息。接着,异步读取文件内容到一个String
变量中,同样使用await
和?
操作符来处理read_to_string
操作可能产生的错误。
在main
函数中,通过match
语句来处理read_file_async
函数返回的Result
类型。由于read_file_async
是异步函数,所以需要使用await
来等待它完成。如果成功,打印文件内容;如果失败,使用eprintln!
打印错误信息,eyre
提供的错误信息会包含详细的错误原因和位置相关内容。
多个异步函数错误处理
在异步编程中,经常需要同时执行多个异步任务,Tokio
提供了join!
和try_join!
宏来处理这种情况。join!
会等待所有任务完成,而try_join!
会在其中一个任务出现错误时立即返回错误。
use tokio::task;
use eyre::Result;
async fn task1() -> Result<i32> {Ok(1)
}
async fn task2() -> Result<i32> {Err(eyre::eyre!("Task 2 error"))
}
#[tokio::main]
async fn main() {let result = task::try_join!(task1(), task2()).map_err(|e| {eprintln!("Error in tasks: {}", e);e});match result {Ok((res1, res2)) => println!("Results: {}, {}", res1, res2),Err(e) => {}}
}
定义了两个异步任务task1
和task2
,其中task1
返回成功的结果,task2
返回一个错误。
在main
函数中,使用try_join!
宏来同时执行这两个任务。如果task2
出现错误,try_join!
会立即返回错误,并且map_err
函数用于处理这个错误,在这里打印错误信息。通过match
语句来处理try_join!
的结果。如果成功,打印两个任务的结果;如果失败,因为已经在map_err
中处理了错误信息,所以这里的Err
分支可以为空。eyre
帮助我们在处理多个异步任务的错误时,提供清晰的错误信息和方便的错误处理机制。
eyre与anyhow的区别
eyre
和anyhow
都是 Rust 语言中用于简化错误处理的库。它们的主要目标是提供一种更方便的方式来处理错误,尤其是在处理复杂的Result
和Option
类型时,减少样板代码,并且能够提供比较友好的错误信息。
在许多常见的 Rust 应用场景中,如文件读取、网络请求、数据库操作等,当需要处理可能出现错误的操作时,这两个库都可以发挥作用。例如,在处理文件读取可能出现的IoError
,或者网络请求返回的错误码时,它们都可以帮助开发者更高效地处理这些错误情况。
eyre 特点
eyre
提供了更丰富的错误上下文信息。它可以包含诸如发生错误的文件名、行号等详细信息。当一个错误在复杂的函数调用栈中传播时,eyre
能够很好地追踪这些信息,并且在报告错误时提供完整的上下文。例如,在一个多层嵌套的函数调用中,eyre
可以精确地指出错误是在哪个文件的哪一行的哪个函数调用中产生的。
eyre
在与其他库集成时,可能需要更多的适配工作,尤其是当涉及到与其他具有复杂错误处理机制的库一起使用时。因为eyre
本身的错误类型和处理方式比较丰富,所以在集成过程中可能需要编写一些转换代码来确保不同库之间的错误能够顺利地传递和处理。
anyhow特点
anyhow
相对来说更侧重于简单地将错误信息进行包装和传播。它的错误信息主要是基于开发者提供的错误描述或者从底层错误类型转换而来的简单文本信息。虽然它也能够有效地传播错误,但在提供详细的错误发生位置等上下文信息方面不如eyre
。
anyhow
由于其简单的错误包装方式,在与其他库集成时通常更加方便。它可以很容易地将其他库产生的错误转换为anyhow::Error
类型,并且在整个应用程序的不同层次之间传递这些错误,而不需要过多复杂的适配代码。例如,在一个使用多个第三方库的项目中,anyhow
可以快速地将这些库产生的各种错误统一起来进行处理。
相关文章:

Rust eyre 错误处理实战教程
在《Rust 错误处理库: thiserror 和 anyhow》中我们介绍了Rust简化处理错误策略,本文解释eyre错误处理库,并通过多个实际示例进行说明,最后于anyhow库进行对比,让你更好理解其应用场景。 eyre是一个用于 Rust 的错误处理库&#x…...
面试小札:JVM虚拟机
1. 定义与基本概念 - JVM(Java Virtual Machine)即Java虚拟机,是Java程序的运行核心。它是一个虚构出来的计算机,通过在实际的计算机上仿真模拟各种计算机功能来运行Java字节码。字节码是一种中间格式,它使得Java程序能…...
Docker扩容操作(docker总是空间不足)
Docker扩容操作(docker总是空间不足) 1、df二连,一共也就70g,总是占满93%以上。所以需要移动到其他目录上 查看docker镜像和容器存储目录的空间大小 du -sh /var/lib/docker/2、停止docker服务 systemctl stop docker3、首先创建目录并迁移 # 首先创…...

数字图像处理(4):FPGA中的定点数、浮点数
(1)定点数:小数点固定在数据的某一位置的数,可以分为定点整数和定点小数和普通定点数。定点数广泛应用于数字图像处理(图像滤波、图像缩放)和数字信号处理(如FFT、定点卷积)中。 定…...
毕昇入门学习
schemas.py 概述 这段代码主要定义了一系列基于 Pydantic 的数据模型(BaseModel),用于数据验证和序列化,通常用于构建 API(如使用 FastAPI)。这些模型涵盖了用户认证、聊天消息、知识库管理、模型配置等多…...
2411C++,学习C++提示4
结构绑定 auto [first, ...ts] std::tuple{1, 2 ,3};assert(1 first);浮点作为非类型模板参数 template<double Value> constexpr auto value Value;int main() {std::cout << value<4.2>; // prints 4.2 }template<double... Vl1s, double... Vl2s&g…...
STM32-- 看门狗--介绍、使用场景、失效场景
STM32 中的看门狗(Watchdog Timer,简称 WDG)有两种主要类型:独立看门狗(IWDG) 和 窗口看门狗(WWDG)。它们的喂狗机制各有特点,主要区别如下: 1. 独立看门狗&a…...

【赵渝强老师】PostgreSQL的数据库
PostgreSQL的逻辑存储结构主要是指数据库中的各种数据库对象,包括:数据库集群、数据库、表、索引、视图等等。所有数据库对象都有各自的对象标识符oid(object identifiers),它是一个无符号的四字节整数,相关对象的oid都…...
linux安全管理-会话安全
文章目录 1 设置命令行界面超时退出2 配置终端登录失败策略3 配置 SSH 登录失败策略 1 设置命令行界面超时退出 1、检查内容 检查操作系统是否设置命令行界面超时退出。 2、配置要求 操作系统设置命令行界面超时退出。 3、配置方法 配置命令行界面超时时间,编辑/et…...
Ubuntu监视显卡占用情况
在终端中运行 watch -n 0.5 nvidia-smi【以下内容由大模型生成】 watch -n 0.5 nvidia-smi 是一个组合命令,用于在 Linux 终端中定期执行 nvidia-smi 命令并显示其输出。让我们分解一下这个命令的各个部分: watch: watch 是一个用于定期执行其他命令并显…...

学成在线day06
上传视屏 断点续传 通常视频文件都比较大,所以对于媒资系统上传文件的需求要满足大文件的上传要求。http协议本身对上传文件大小没有限制,但是客户的网络环境质量、电脑硬件环境等参差不齐,如果一个大文件快上传完了网断了没有上传完成&…...

Mac安装及合规无限使用Beyond Compare
文章目录 Beyond CompareBeyond Compare简介Beyond Compare安装Beyond Compare到期后继续免费使用 Beyond Compare Beyond Compare简介 Beyond Compare 是一款由 Scooter Software 开发的文件和文件夹比较工具。它主要用于对比两个文件或文件夹之间的差异,并支持文…...

【青牛科技】2K02 电动工具专用调速电路芯片描述
概述: 2K02 是电动工具专用调速电路。内置稳压电路,温度系数好,可以调节输出频率以及占空比的振荡输出,广泛的应用于小型电钻,割草机等工具。 主要特点: ● 电源电压范围宽 ● 占空比可调 ● 温度系数好 …...

基于SpringBoot实现的民宿管理系统(代码+论文)
🎉博主介绍:Java领域优质创作者,阿里云博客专家,计算机毕设实战导师。专注Java项目实战、毕设定制/协助 📢主要服务内容:选题定题、开题报告、任务书、程序开发、项目定制、论文辅导 💖精彩专栏…...

安装QT6.8(MSVC MinGW)+QT webengine+QT5.15.2
本篇主要针对只使用过QT5的qmake,没有用过MSVC,VS的老同学。 建议一部分一部分安装,全部勾选安装遇到问题会中断,前功尽弃。 我自己需要的是QT5,编出的软件用在公司设备上。 QT6:建议也安装学习…...
MinIO常见操作及Python实现对象的增删改查
MinIO常见操作 MinIO是一个高性能的开源对象存储服务,它兼容Amazon S3云存储服务API。在MinIO中,常见的操作包括: 存储桶操作: 创建、列出、获取信息、删除存储桶。 对象操作: 上传、下载、列出、删除对象。 权限管理&…...
网络编程中的字节序函数htonl()、htons()、ntohl()和ntohs()
目录 1,网络字节序和主机字节序 2. 函数的具体作用 2.1,htonl(Host to Network Long) 2.2,htons(Host to Network Short) 2.3,ntohl(Network to Host Long) 2.4,ntohs(Network to Host Sho…...

【dvwa靶场:File Upload系列】File Upload低-中-高级别,通关啦
目录 一、low级别,直接上传木马文件 1.1、准备一个木马文件 1.2、直接上传木马文件 1.3、访问木马链接 1.4、连接蚁剑 二、medium级别:抓包文件缀名 2.1、准备一个木马文件,修改后缀名为图片的后缀名 2.2、上传文件,打开burpSuite&…...

RHCE NFS
RHCE NFS 1.11. 2 NFS 主机名格式1.3 NFS 服务器配置1.3.1 /etc/exports 配置文件1.3.1.1 导出条目1.3.1.2 默认选项1.3.1.3 默认和覆盖选项 1.4 启动 NFS 服务器1.5 练习1.5.1 配置 NFS 服务器和客户端挂载1.5.2 配置autofs自动挂载(需要时才挂载) 1.6 …...

【数据结构】ArrayList与顺序表
ArrayList与顺序表 1.线性表2.顺序表2.1 接口的实现 3. ArrayList简介4. ArrayList使用4.2 ArrayList常见操作4.3 ArrayList的遍历4.4 ArrayList的扩容机制 5. ArrayList的具体使用5.1 杨辉三角5.2 简单的洗牌算法 6. ArrayList的问题及思考 【本节目标】 线性表顺序表ArrayLis…...

XCTF-web-easyupload
试了试php,php7,pht,phtml等,都没有用 尝试.user.ini 抓包修改将.user.ini修改为jpg图片 在上传一个123.jpg 用蚁剑连接,得到flag...

微信小程序之bind和catch
这两个呢,都是绑定事件用的,具体使用有些小区别。 官方文档: 事件冒泡处理不同 bind:绑定的事件会向上冒泡,即触发当前组件的事件后,还会继续触发父组件的相同事件。例如,有一个子视图绑定了b…...

突破不可导策略的训练难题:零阶优化与强化学习的深度嵌合
强化学习(Reinforcement Learning, RL)是工业领域智能控制的重要方法。它的基本原理是将最优控制问题建模为马尔可夫决策过程,然后使用强化学习的Actor-Critic机制(中文译作“知行互动”机制),逐步迭代求解…...

【JavaEE】-- HTTP
1. HTTP是什么? HTTP(全称为"超文本传输协议")是一种应用非常广泛的应用层协议,HTTP是基于TCP协议的一种应用层协议。 应用层协议:是计算机网络协议栈中最高层的协议,它定义了运行在不同主机上…...

基于ASP.NET+ SQL Server实现(Web)医院信息管理系统
医院信息管理系统 1. 课程设计内容 在 visual studio 2017 平台上,开发一个“医院信息管理系统”Web 程序。 2. 课程设计目的 综合运用 c#.net 知识,在 vs 2017 平台上,进行 ASP.NET 应用程序和简易网站的开发;初步熟悉开发一…...

vscode(仍待补充)
写于2025 6.9 主包将加入vscode这个更权威的圈子 vscode的基本使用 侧边栏 vscode还能连接ssh? debug时使用的launch文件 1.task.json {"tasks": [{"type": "cppbuild","label": "C/C: gcc.exe 生成活动文件"…...
线程与协程
1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指:像函数调用/返回一样轻量地完成任务切换。 举例说明: 当你在程序中写一个函数调用: funcA() 然后 funcA 执行完后返回&…...

2025盘古石杯决赛【手机取证】
前言 第三届盘古石杯国际电子数据取证大赛决赛 最后一题没有解出来,实在找不到,希望有大佬教一下我。 还有就会议时间,我感觉不是图片时间,因为在电脑看到是其他时间用老会议系统开的会。 手机取证 1、分析鸿蒙手机检材&#x…...
C++中string流知识详解和示例
一、概览与类体系 C 提供三种基于内存字符串的流,定义在 <sstream> 中: std::istringstream:输入流,从已有字符串中读取并解析。std::ostringstream:输出流,向内部缓冲区写入内容,最终取…...
【RockeMQ】第2节|RocketMQ快速实战以及核⼼概念详解(二)
升级Dledger高可用集群 一、主从架构的不足与Dledger的定位 主从架构缺陷 数据备份依赖Slave节点,但无自动故障转移能力,Master宕机后需人工切换,期间消息可能无法读取。Slave仅存储数据,无法主动升级为Master响应请求ÿ…...