手机如何做微商城网站,国外学做咖啡的网站,石狮住房和城乡建设局网站,深圳开发网站开发Rust 将错误分为两大类#xff1a;可恢复的#xff08;recoverable#xff09;和 不可恢复的#xff08;unrecoverable#xff09;错误。对于一个可恢复的错误#xff0c;比如文件未找到的错误#xff0c;我们很可能只想向用户报告问题并重试操作。不可恢复的错误总是 b…Rust 将错误分为两大类可恢复的recoverable和 不可恢复的unrecoverable错误。对于一个可恢复的错误比如文件未找到的错误我们很可能只想向用户报告问题并重试操作。不可恢复的错误总是 bug 出现的征兆比如试图访问一个超过数组末端的位置因此我们要立即停止程序。 Rust 的错误处理主要通过 Result 和 Option 类型来实现。Result 类型表示一个操作可能会成功返回 Ok或失败返回 Err而 Option 类型表示一个值可能存在Some或不存在None。
一、不可恢复异常
C 语言中尝试读取数据结构之后的值是未定义行为undefined behavior。你会得到任何对应数据结构中这个元素的内存位置的值甚至是这些内存并不属于这个数据结构的情况。这被称为 缓冲区溢出buffer overread并可能会导致安全漏洞比如攻击者可以像这样操作索引来读取储存在数据结构之后不被允许的数据。为了保护程序远离这类漏洞如果尝试读取一个索引不存在的元素Rust 会停止执行并拒绝继续。
fn main() {let v vec![1, 2, 3];v[99];
}运行产生异常
thread main panicked at src\main.rs:246:6:
index out of bounds: the len is 2 but the index is 99二、可恢复异常
使用result或者option进行异常处理。
fn divide(a: f64, b: f64) - Resultf64, String {if b 0.0 {Err(除数不能为0.to_string())} else {Ok(a / b)}
}fn main() {let result divide(10.0, 2.0);match result {Ok(value) println!(结果 {}, value),Err(err) println!(错误 {}, err),}
}
fn find_index(vec: Veci32, target: i32) - Optionusize {for (index, value) in vec.iter().enumerate() {if value target {return Some(index);}}None
}fn main() {let vec vec![1, 2, 3, 4, 5];let index find_index(vec, 3);match index {Some(value) println!(找到目标值索引为 {}, value),None println!(未找到目标值),}
}
match已经可以判断是否异常但是比较长还有是通过unwarp的方式如果result的值是Ok,那么会返回结果如果是Errunwrap会调用panic!
use std::fs::File;
fn main() {let greeting_file File::open(hello.txt).unwrap();
}产生异常时
thread main panicked at called Result::unwrap() on an Err value: Os {
code: 2, kind: NotFound, message: No such file or directory },
src/main.rs:4:49也可以通过expect方法去自定义异常输出。
三、错误的传播
从一个方法里面抛出了错误外层调用的方法怎么捕获异常类似java的throw try catch
fn read_username_from_file() - ResultString, io::Error {let username_file_result File::open(hello.txt);let mut username_file match username_file_result {Ok(file) file,Err(e) return Err(e),};let mut username String::new();match username_file.read_to_string(mut username) {Ok(_) Ok(username),Err(e) Err(e),}
}也可以使用简写: ? 运算符
use std::fs::File;
use std::io::{self, Read};
fn read_username_from_file() - ResultString, io::Error {let mut username_file File::open(hello.txt)?;let mut username String::new();username_file.read_to_string(mut username)?;Ok(username)
}还可以进一步缩短
use std::fs::File;
use std::io::{self, Read};
fn read_username_from_file() - ResultString, io::Error {let mut username String::new();File::open(hello.txt)?.read_to_string(mut username)?;Ok(username)
}哪里可以使用 ? 运算符 ? 运算符只能被用于返回值与 ? 作用的值相兼容的函数。因为 ? 运算符被定义为从函数中提早返回一个值这与上面写的 match 表达式有着完全相同的工作方式。