? 的其他用途
注意在前面的例子中,我们对调用 parse 的直接反应是将库错误通过 map 转换为一个装箱的错误:
.and_then(|s| s.parse::<i32>())
.map_err(|e| e.into())
由于这是一个简单且常见的操作,如果能够省略就会很方便。可惜的是,由于 and_then 不够灵活,所以无法实现这一点。不过,我们可以使用 ? 来替代。
之前我们将 ? 解释为 unwrap 或 return Err(err)。这只是大致正确。实际上,它的含义是 unwrap 或 return Err(From::from(err))。由于 From::from 是不同类型之间的转换工具,这意味着如果你在错误可转换为返回类型的地方使用 ?,它将自动进行转换。
在这里,我们使用 ? 重写了前面的例子。当为我们的错误类型实现 From::from 后,map_err 就不再需要了:
use std::error;
use std::fmt;
// 将别名改为使用 `Box<dyn error::Error>`。
type Result<T> = std::result::Result<T, Box<dyn error::Error>>;
#[derive(Debug)]
struct EmptyVec;
impl fmt::Display for EmptyVec {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "无效的第一个待加倍项")
}
}
impl error::Error for EmptyVec {}
// 结构与之前相同,但不再链式处理所有的 `Result` 和 `Option`,
// 而是使用 `?` 立即获取内部值。
fn double_first(vec: Vec<&str>) -> Result<i32> {
let first = vec.first().ok_or(EmptyVec)?;
let parsed = first.parse::<i32>()?;
Ok(2 * parsed)
}
fn print(result: Result<i32>) {
match result {
Ok(n) => println!("第一个数的两倍是 {}", n),
Err(e) => println!("错误:{}", e),
}
}
fn main() {
let numbers = vec!["42", "93", "18"];
let empty = vec![];
let strings = vec!["tofu", "93", "18"];
print(double_first(numbers));
print(double_first(empty));
print(double_first(strings));
}
现在代码变得相当简洁了。与原来使用 panic 的版本相比,这种方法非常类似于用 ? 替换 unwrap 调用,只是返回类型变成了 Result。因此,需要在顶层对结果进行解构。