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…...
在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...
代理篇12|深入理解 Vite中的Proxy接口代理配置
在前端开发中,常常会遇到 跨域请求接口 的情况。为了解决这个问题,Vite 和 Webpack 都提供了 proxy 代理功能,用于将本地开发请求转发到后端服务器。 什么是代理(proxy)? 代理是在开发过程中,前端项目通过开发服务器,将指定的请求“转发”到真实的后端服务器,从而绕…...
Bean 作用域有哪些?如何答出技术深度?
导语: Spring 面试绕不开 Bean 的作用域问题,这是面试官考察候选人对 Spring 框架理解深度的常见方式。本文将围绕“Spring 中的 Bean 作用域”展开,结合典型面试题及实战场景,帮你厘清重点,打破模板式回答,…...
LangFlow技术架构分析
🔧 LangFlow 的可视化技术栈 前端节点编辑器 底层框架:基于 (一个现代化的 React 节点绘图库) 功能: 拖拽式构建 LangGraph 状态机 实时连线定义节点依赖关系 可视化调试循环和分支逻辑 与 LangGraph 的深…...
DBLP数据库是什么?
DBLP(Digital Bibliography & Library Project)Computer Science Bibliography是全球著名的计算机科学出版物的开放书目数据库。DBLP所收录的期刊和会议论文质量较高,数据库文献更新速度很快,很好地反映了国际计算机科学学术研…...
Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析
Java求职者面试指南:Spring、Spring Boot、Spring MVC与MyBatis技术解析 一、第一轮基础概念问题 1. Spring框架的核心容器是什么?它的作用是什么? Spring框架的核心容器是IoC(控制反转)容器。它的主要作用是管理对…...
LLaMA-Factory 微调 Qwen2-VL 进行人脸情感识别(二)
在上一篇文章中,我们详细介绍了如何使用LLaMA-Factory框架对Qwen2-VL大模型进行微调,以实现人脸情感识别的功能。本篇文章将聚焦于微调完成后,如何调用这个模型进行人脸情感识别的具体代码实现,包括详细的步骤和注释。 模型调用步骤 环境准备:确保安装了必要的Python库。…...
消防一体化安全管控平台:构建消防“一张图”和APP统一管理
在城市的某个角落,一场突如其来的火灾打破了平静。熊熊烈火迅速蔓延,滚滚浓烟弥漫开来,周围群众的生命财产安全受到严重威胁。就在这千钧一发之际,消防救援队伍迅速行动,而豪越科技消防一体化安全管控平台构建的消防“…...
云原生安全实战:API网关Envoy的鉴权与限流详解
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、基础概念 1. API网关 作为微服务架构的统一入口,负责路由转发、安全控制、流量管理等核心功能。 2. Envoy 由Lyft开源的高性能云原生…...
CTF show 数学不及格
拿到题目先查一下壳,看一下信息 发现是一个ELF文件,64位的 用IDA Pro 64 打开这个文件 然后点击F5进行伪代码转换 可以看到有五个if判断,第一个argc ! 5这个判断并没有起太大作用,主要是下面四个if判断 根据题目…...
