10.错误处理

标题

  • 一、概述
  • 二、panic!与不可恢复错误
    • 2.1 出错时
    • 2.2 示例
    • 2.3 panic!的 backtrace
  • 三、Result 与可恢复的错误
    • 3.1 引入
    • 3.2 错误示例
    • 3.3 传播错误
      • 1)概念
      • 2)传播错误示例
      • 3)传播错误的简写:?运算符

一、概述

  • Rust将错误分成两类:可恢复错误不可恢复错误
  • 可恢复错误主要是未找到文件等,不可恢复错误通常会造成程序panic;
  • 可恢复错误主要用Result<T, E>表示,不可恢复错误用panic!

二、panic!与不可恢复错误

2.1 出错时

当使用panic!宏表示不可恢复错误时,程序通过做如下操作:

  1. 打印错误信息;
  2. 展开并清理栈数据;
  3. 退出;

panic时,如果要求程序直接终止,可以打开Cargo.toml文件,添加如下配置信息

[profile.release]
panic = 'abort'

这样也可以减少应用程序大小。

2.2 示例

基本panic!

fn main() {panic!("Error");
}

程序运行如下

thread 'main' panicked at src\main.rs:2:5:
Error
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
error: process didn't exit successfully: `target\debug\testrust.exe` (exit code: 101)
PS G:\rustobject\testrust> cargo runFinished `dev` profile [unoptimized + debuginfo] target(s) in 0.02sRunning `target\debug\testrust.exe`
thread 'main' panicked at src\main.rs:2:5:
Error
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
error: process didn't exit successfully: `target\debug\testrust.exe` (exit code: 101)
  • 表明panic的位置位于src\main.rs文件第2行第5个字符
  • 当前显示的是代码中的panic!位置;
  • 一般情况下,panic!可能会出现在所调用的代码中;错误信息报告的文件名和行号可能指向别人代码中的panic!宏调用;
  • 可以使用backtrace追踪;

2.3 panic!的 backtrace

数组越界的出错

fn main() {let x = vec![1, 2, 3, 4];x[4];
}

报错信息如下
在这里插入图片描述

  • 根据提示,使用$env:RUST_BACKTRACE=1; cargo run命令重新运行程序,结果如下
    在这里插入图片描述
  • 上图中的6:指明了代码出错的地方;

三、Result 与可恢复的错误

3.1 引入

  • 大部分错误并没有严重到需要程序完全停止执行;
  • 比如说要打开的文件不存在或创建文件时没有权限;
  • 因此引入了Result枚举
enum Result<T, E> {Ok(T),Err(E),
}

在Result枚举里:

  • T 和 E 是泛型类型参数
  • T 代表成功时返回的Ok成员中的数据的类型;
  • E 代表失败时返回的Err成员中的错误的类型;

3.2 错误示例

以打开文件示例

use std::fs::File;fn main() {let f  = File::open("hello.txt");
}
  • 这是一个简单的打开文件测试,把鼠标放到open上面会显示它的详细信息,下面的灰色部分显示了open函数的返回值(复制时不在,只能截图看)
    在这里插入图片描述
  • 由此可见,Result<T, E>中的T是成功值打开文件的文件句柄std::fs::File,E 被用在失败值上的;
  • 双击红框的部分会自动插入完整类型(这时就可以复制了),这样可以将返回值看的更加清楚;
    在这里插入图片描述
  • 当open成功的情况下,变量 f 的值将会是一个包含文件句柄的实例;
  • 在失败的情况下,f 的值会是一个包含更多关于出现了何种错误信息的实例;

下面是一个加入判断的例子

use std::fs::File;fn main() {let f  = File::open("hello.txt");let f = match f {Ok(file) => file,Err(error) => {panic!("Problem opening the file: {:?}", error)},};
}

报错信息如下
在这里插入图片描述

  • 当打开文件成功时,将文件句柄返回给f;
  • 当打开文件失败时,输出提示信息;

仔细看上图红框的部分,有两个值是code和kind,说明可以对不同的错误值进行分类。

错误匹配

use std::fs::File;
use std::io::ErrorKind;fn main() {let f  = File::open("hello.txt");let f = match f {Ok(file) => file,Err(error) => match error.kind() {ErrorKind::NotFound => match File::create("hello.txt") {Ok(fc) => fc,Err(e) => panic!("Problem creating the file: {:?}", e),},other_error => panic!("Problem opening the file: {:?}", other_error),},};
}

代码将错误类型进行分类

  • 没有找到文件:创建新文件;
  • 其它问题(权限问题(PermissionDenied)等):提示无法打开文件;

打开失败时的另一种写法:unwrap和expect

  • Result<T, E>类型定义了很多辅助方法处理各种情况,其中之一是unwrap,中文叫拆包
  • 功能类似于match语句:Result值是成员Ok,unwrap返回Ok中的值,否则会调用panic!

使用unwrap

use std::fs::File;fn main() {let f = File::open("hello.txt").unwrap();
}

运行结果
在这里插入图片描述
使用expect

use std::fs::File;fn main() {let f = File::open("hello.txt").expect("系统找不到hello.txt文件");
}

运行结果
在这里插入图片描述

  • 这个错误信息是代码里提供的,因此更容易找到;

3.3 传播错误

1)概念

传播错误是指:当编写一个需要先调用一些可能会失败的操作的函数时,除了在这个函数中处理错误外,还可以选择让调用者知道这个错误并决定该如何处理。

2)传播错误示例

use std::io;
use std::io::Read;
use std::fs::File;fn read_username_from_file() -> Result<String, io::Error> {let f = File::open("hello.txt");let mut f = match f {Ok(file) => file,Err(e) => return Err(e),};let mut s = String::new();match f.read_to_string(&mut s) {Ok(_) => Ok(s),Err(e) => Err(e),}
}
  • 这是一个读文件内容(用户名)的函数;
  • 里面的openread_to_string函数都返回Result;
  • 如果文件不存在或者其他原因不能读取,函数会被错误返回给调用它的代码;
  • 代码编写人员无法确定调用者具体会如何做,这段代码只负责将成功或失败的消息向上传播;
  • 这种传播模式Rust提供了?运算符来简化编写流程;

3)传播错误的简写:?运算符

下面展示了另一种read_username_from_file的写法,使用?功能后明显显得更加简洁

  • 返回Result的函数在函数之后加上?运算符;
  • 如果返回的值是Ok,这个表达式将会返回Ok中的值而程序将继续执行;
  • 如果返回值是Err,则Err将作为整个函数的返回值返回给调用者;
use std::io;
use std::io::Read;
use std::fs::File;fn read_username_from_file() -> Result<String, io::Error> {let mut f = File::open("hello.txt")?;let mut s = String::new();f.read_to_string(&mut s)?;Ok(s)
}
  • 可以在?之后直接使用链式方法进一步缩短代码
use std::io;
use std::io::Read;
use std::fs::File;fn read_username_from_file() -> Result<String, io::Error> {let mut s = String::new();File::open("hello.txt")?.read_to_string(&mut s)?;Ok(s)
}
  • 打开文件、新建一个String、 读取文件内容并放入String并返回String,这套联招可以用下面的代码直接搞定
use std::io;
use std::fs;fn read_username_from_file() -> Result<String, io::Error> {fs::read_to_string("hello.txt")
}

注意:只能在返回值时Result的函数中使用?运算符

将?运算符移动到main函数中

use std::fs::File;fn main() {let f = File::open("hello.txt")?;
}

编译结果如下
在这里插入图片描述

  • main函数的返回值是有限制的;
  • main函数的一个有效的返回值是 () ;
  • 出于方便,另一个有效的返回值是 Result<T, E>

修改成下面这样,就可以通过编译

use std::fs::File;
use std::error::Error;fn main() -> Result<(), Box<dyn Error>> {let f = File::open("hello.txt")?;Ok(())
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/27559.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

springboot网上书店管理系统-计算机毕业设计源码03780

摘 要 网上书店管理系统采用B/S结构、java开发语言、以及Mysql数据库等技术。系统主要分为管理员和用户两部分&#xff0c;管理员管理主要功能包括&#xff1a;首页、站点管理&#xff08;轮播图&#xff09;用户管理&#xff08;管理员、注册用户&#xff09;内容管理&#x…

深入理解ReentrantLock

深入理解ReentrantLock 在Java并发编程中&#xff0c;锁&#xff08;Lock&#xff09;是控制多个线程对共享资源访问的重要工具。虽然Synchronized关键字是实现锁的常用方式&#xff0c;但它在功能上比较有限。ReentrantLock是java.util.concurrent.locks包中提供的一个更加灵…

FreeRTOS移植:STM32L476 nucleo-L476RG 开发板《02》

系列文章 FreeRTOS移植&#xff1a;STM32L476 nucleo-L476RG 开发板《01》 说明 上一篇 FreeRTOS移植&#xff1a;STM32L476 nucleo-L476RG 开发板《01》 主要讲了一下如何快速搭建一个 STM32 裸机工程&#xff0c;其实 STM32CubeMX 可以生成 FreeRTOS 的工程&#xff0c;这就…

Linux host映射 设置主机名并通过主机名找到指定系统

一、windows ping linux 1.windows进入到 C:\Windows\System32\drivers\etc\hosts 内&#xff0c;使用edit with notepad打开hosts文件&#xff0c;在下面添加需要寻找的ip以及其主机名。该ip以及主机名即linux的一致。需要查看linux主机名的在终端使用 hostname进行查看&#…

spring boot3登录开发-邮箱登录/注册接口实现

⛰️个人主页: 蒾酒 &#x1f525;系列专栏&#xff1a;《spring boot实战》 &#x1f30a;山高路远&#xff0c;行路漫漫&#xff0c;终有归途 目录 写在前面 上文衔接 内容简介 功能分析 所需依赖 邮箱验证登录/注册实现 1.创建交互对象 2.登录注册业务逻辑实…

JDK17 你的下一个白月光

JDK版本升级的非常快&#xff0c;现在已经到JDK20了。JDK版本虽多&#xff0c;但应用最广泛的还得是JDK8&#xff0c;正所谓“他发任他发&#xff0c;我用Java8”。 但实际情况却不是这样&#xff0c;越来越多的java工程师拥抱 JDK17&#xff0c;于是了解了一下 JDK17新语法&a…

Star-CCM+自动网格执行方法与设置技巧

在Star中进行一个仿真项目时,有时会创建多个自动网格。网格创建结束后需要执行。在Star中,网格执行可以分为三种。分别是:单独执行操作;多个执行操作;全部执行操作。接下来将三种执行操作的方法与步骤进行介绍。 其次,如果不习惯用自定义控制网格,有时在一个项目中就会…

大模型+人工智能:重塑地方志管理的新力量

前言 在科技日新月异的今天&#xff0c;人工智能&#xff08;AI&#xff09;正以前所未有的速度渗透到各个领域&#xff0c;改变着我们的工作和生活方式。特别是在地方志管理这一领域&#xff0c;大模型和人工智能的结合正在开启一场深刻的变革。今天&#xff0c;就让我们一起…

【复旦邱锡鹏教授《神经网络与深度学习公开课》笔记】梯度的反向传播算法

矩阵微积分&#xff08;Matrix Calculus&#xff09; 在开始之前&#xff0c;需要先了解矩阵微积分的一些计算规则。 首先&#xff0c;对于矩阵微积分的表示&#xff0c;通常由两种符号约定&#xff1a; 分母布局 标量关于向量的导数为列向量 向量关于标量的导数为行向量 N维…

LDR6500:手机电脑拓展坞转接器方案的卓越之选

随着科技的飞速发展&#xff0c;手机和电脑已成为我们日常生活中不可或缺的工具。然而&#xff0c;它们的接口有限&#xff0c;经常难以满足我们多样化的需求。这时&#xff0c;一款高效、稳定的拓展坞转接器就显得尤为重要。LDR6500&#xff0c;作为乐得瑞科技精心研发的USB P…

【计算机视觉(10)】

基于Python的OpenCV基础入门——图像滤波去噪 图像滤波去噪均值滤波中值滤波高斯滤波双边滤波方框滤波图像滤波去噪代码实现及其效果图 图像滤波去噪 图像滤波去噪是一种图像处理方法&#xff0c;它通过应用滤波器来减少或消除图像中的噪声。噪声是图像中不希望的、无用的、干…

安装sqlserver2022 express

1、下载 SQL Server 下载 | Microsoft 双击sql2022-ssei-expr 2、安装 下载完成以后&#xff0c;将会出现以下对话框 &#xff1a; 点击【全新SQL Server独立安装或向现有安全添加功能】 下一步&#xff0c;下一步&#xff1a; 下一步&#xff1a; 下一步&#xff0c;这里我…

【LeetCode:2779. 数组的最大美丽值 + 排序 + 二分】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

【docker】如何解决artalk的跨域访问问题

今天折腾halo的时候&#xff0c;发现artalk出现跨域访问报错&#xff0c;内容如下。 Access to fetch at https://artk.musnow.top/api/stat from origin https://halo.musnow.top has been blocked by CORS policy: The Access-Control-Allow-Origin header contains multipl…

Kotlin编程实践-【Java如何调用Kotlin中带默认值参数的函数】

问题 如果你有一个带有默认参数值的 Kotlin 函数&#xff0c;如何从 Java 调用它而无须为每个参数显式指定值&#xff1f; 方案 为函数添加注解JvmOverloads。 也就是为Java添加重载方法&#xff0c;这样Java调用Kotlin的方法时就不用传递全部的参数了。 示例 在 Kotlin …

企业环保创A标准

在环保日益受到重视的今天&#xff0c;企业如何有效地进行环保管理&#xff0c;提高自身的环保水平&#xff0c;已成为一个不可忽视的议题。而企业环保创A标准&#xff0c;正是为了评估企业的环保水平和环保管理能力而制定的一项重要评价标准。朗观视觉小编将详细解析企业环保创…

XILINX 7系列XDMA使用_IP核介绍以及工程搭建

文章目录 一、XDMA IP核1.1、接口说明1.2、配置页说明 二、XDMA工程搭建2.1、BD搭建2.2 Linux下XDMA驱动安装2.3 Linux下使用XDMA进行数据传输 一、XDMA IP核 1.1、接口说明 sys_clk&#xff1a;主机给PCIE提供的时钟信号&#xff0c;通过原理图查看 sys_rst_n&#xff1a;主机…

【已解决】引入 element 组件无法使用编译错误 ERROR Failed to compile with 1 error

如果大家使用这个vue 配合 element 框架不熟练&#xff0c;当你顺利按照文档安装好 vue 和 element 的时候想要使用element 的组件时候确无法展示出来&#xff0c;甚至报错。不妨看看是不是这个问题&#xff0c; 1.首先使用element 的时候&#xff0c;前提是把必须要的 elemen…

TCGAbiolinks包学习

TCGAbiolinks 写在前面学习目的GDCquery GDCdownload GDC prepare中间遇到的报错下载蛋白质数据 写在前面 由于别人提醒我TCGA的数据可以利用TCGAbiolinks下载并处理&#xff0c;所以我决定阅读该包手册&#xff0c;主要是该包应该是有更新的&#xff0c;我看手册进行更新了&…

法国恐脱欧、陷金融危机!股指本周跌6.2%,创三年多最大跌幅

内容提要 法国财政部长警告称&#xff0c;左翼政党联盟若上台可能导致法国脱欧&#xff0c;而且无论极右翼还是左翼上台&#xff0c;都可能导致法国爆发金融危机。由于政坛风险高企&#xff0c;法国股市周五延续跌势&#xff0c;本周已经抹掉2100亿美元市值&#xff0c;几乎回…