イテレーションでResultを表す
Iter::mapは失敗する可能性があります。例えば:
fn main() { let strings = vec!["tofu", "93", "18"]; let numbers: Vec<_> = strings .into_iter() .map(|s| s.parse::<i32>()) .collect(); println!("Results: {:?}", numbers); }
これを処理する方法を見ていきましょう。
filter_map()で失敗したものを無視する
filter_mapはNoneである結果をフィルタにかける。
fn main() { let strings = vec!["tofu", "93", "18"]; let numbers: Vec<_> = strings .into_iter() .map(|s| s.parse::<i32>()) .filter_map(Result::ok) .collect(); println!("Results: {:?}", numbers); }
collect()でコードを失敗にする
ResultはFromIterを実装しているため、Result型のベクター(Vec<Result<T, E>>)
をベクター型のResult(Result<Vec<T>, E>)に変換することができます。
そして、Result::Errが見つかれば処理を終了できます。
fn main() { let strings = vec!["tofu", "93", "18"]; let numbers: Result<Vec<_>, _> = strings .into_iter() .map(|s| s.parse::<i32>()) .collect(); println!("Results: {:?}", numbers); }
Optionでも同じことができます。
すべての利用可能な値を集めてpartition()で失敗を振り分ける。
fn main() { let strings = vec!["tofu", "93", "18"]; let (numbers, errors): (Vec<_>, Vec<_>) = strings .into_iter() .map(|s| s.parse::<i32>()) .partition(Result::is_ok); println!("Numbers: {:?}", numbers); println!("Errors: {:?}", errors); }
結果を見たら、これがまだResultで包まれていることに気付くでしょう。
もう少し典型コードが必要です。
fn main() { let strings = vec!["tofu", "93", "18"]; let (numbers, errors): (Vec<_>, Vec<_>) = strings .into_iter() .map(|s| s.parse::<i32>()) .partition(Result::is_ok); let numbers: Vec<_> = numbers.into_iter().map(Result::unwrap).collect(); let errors: Vec<_> = errors.into_iter().map(Result::unwrap_err).collect(); println!("Numbers: {:?}", numbers); println!("Errors: {:?}", errors); }