? 的其他用途

注意在前面的例子中,我们对调用 parse 的直接反应是将库错误通过 map 转换为一个装箱的错误:

.and_then(|s| s.parse::<i32>())
    .map_err(|e| e.into())

由于这是一个简单且常见的操作,如果能够省略就会很方便。可惜的是,由于 and_then 不够灵活,所以无法实现这一点。不过,我们可以使用 ? 来替代。

之前我们将 ? 解释为 unwrapreturn Err(err)。这只是大致正确。实际上,它的含义是 unwrapreturn 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。因此,需要在顶层对结果进行解构。

另请参阅:

From::from?运算符