rust中没异常机制,这意味着我们不用写出一层层的try...catch
语句。异常处理有两个核心的地方,分别是panic!
和Result<T, E>
泛型。分别总结这两个的核心。
panic!
是一个异常的宏,出现这个,程序会立刻终止,然后通过panic!
的内容打印错误信息。给出代码实例:
fn main() {
panic!("fatal"); // 这里直接终止掉整个程序
}
有些库函数,在调用失败的时候也会产生panic
,为了方便追踪打印日志,可以使用.expect()
的方式添加追踪信息,代码实例:
use std::fs::File;
fn main() {
// 如果失败,在这里直接panic,但是没有错误信息
let f = File::open("hello.txt").unwrap();
// 显示对应的信息
let f1 = File::open("hello.txt").expect("Filed open hello.txt");
}
不是所有的程序都需要panic终止整个进程,可以有Resutl<T, E>
的泛型来处理,这个类似于Option
,可以返回泛型的结果或者是错误的类型,直接给出代码实例:
use std::io;
use std::io::Read;
use std::fs::File;
fn read(filename: &String) -> Result<String, io::Error> {
let f = File::open(filename);
let mut f = match f {
OK(file) => file, // 接住包装的file,但是返回实际的file
Err(e) => return Err(e),
};
let mut s = String::new();
// expression类型,直接作为返回值即可
match f.read_to_string {
OK(_) => OK(s), // 接住任意的传入结果,返回包装的读取结果
Err(e) => Err(e),
}
上述代码稍微有点冗余,rust给出了更为优雅的错误处理链,通过?
来传递错误:
use std::io;
use std::io::Read;
use std::fs::File;
fn read(filename: String) -> Result<String, io::Error> {
let mut s = String::new();
File::open(filename)?.read_to_string(&mut s)?;
OK(s)
}
逐层分析,File::open(filename)?
如果成功,那么返回一个file
类型,然后调用read_to_string
函数继续读取数据;失败的话直接返回对应的错误。下面的read_to_string
同理,不在赘述。