?,rust"/>

首页 > 解决方案 > 如何习惯性地从 Option 转换导致?

问题描述

我是 Rust 新手!从 a 转换Option<T>为 a的最佳方法是Result<T, ()>什么?

TryFrom特征似乎普遍并返回一个结果。流行的 num_traits 的NumCast有很多转换,但它们都返回一个Option<T>. 类似地, Rust 标准库中的 NonZero* 构造函数(例如NonZeroI32)也是如此。然后我注意到 NumCast 实现了 a返回 an所以我想它可能有一种非标准的做事方式,但后来我看到了 NonZero* 实现并质疑了这个想法。from()Option<T>

无论如何,从选项转换为结果似乎很频繁,我还没有找到一个很好的方法。例如:

/// Add common conversion from an i32 to a non-zero i32.
impl TryFrom<Container<i32>> for Container<NonZeroI32> {
  type Error = ();
  fn try_from(container: Container<i32>) -> Result<Self, ()> {
    // NonZeroI32::new() returns an Option not a Result. Try a helper.
    Ok(Self(option_to_result(NonZeroI32::new(container.0))?))
  }
}

/// Helper function to convert from an Option to a Result (both types are
/// foreign and so is From).
fn option_to_result<T>(option: Option<T>) -> Result<T, ()> {
  if let Some(some) = option {
    Ok(some)
  } else {
    Err(())
  }
}
/// Add another common conversion from an i32 to an i16.
impl TryFrom<Container<i32>> for Container<i16> {
  type Error = ();
  fn try_from(container: Container<i32>) -> Result<Self, ()> {
    // NumCast::from() also returns an Option not a Result. Try map_or() instead
    // of the helper.
    Ok(Self(NumCast::from(container.0).map_or(Err(()), |x| Ok(x))?))
  }
}

( Rust Playground中的上述示例。)

这些 NumCast、NonZero* 和 TryFrom 转换似乎很常见,但我的方法感觉很笨拙,就好像我将 Option 和 Result 类型相互竞争一样。我在这些转换中苦苦挣扎,也错过了Option给定类型Result<T,()>感觉相似的基本点。

那么,在 Rust 2018中转换Option<T>为的惯用方式是什么?Result<T,()>

标签: rust

解决方案


Optionok_or方法,可用于此(实际上,对于更广泛的情况,但您的要求也适合这里):

fn option_to_result<T>(option: Option<T>) -> Result<T, ()> {
  option.ok_or(())
}

改装游乐场


推荐阅读