是否可以在不使用匹配的情况下将 Option<Result<T, E>> 转换为 Result<Option<T>, E>?

Is it possible to convert Option<Result<T, E>> to a Result<Option<T>, E> without using match?

我的第一个想法是 map Option,但我不能在闭包内部使用 try!match 语句看起来没有必要,但我不知道如何简化它。

fn example<T, E>(val: Option<Result<T, E>>) -> Result<Option<T>, E> {
    Ok(match val {
        Some(v) => Some(v?),
        None => None
    })
}

您可以使用 Option::map_or():

val.map_or(Ok(None), |v| v.map(Some))

在 Rust 1.33 中,transpose() 是稳定的,所以你可以直接调用它:

fn main() {
    let x: Result<Option<i32>, ()> = Ok(Some(5));
    let y: Option<Result<i32, ()>> = Some(Ok(5));
    assert_eq!(x, y.transpose());
}