//! Traits for writing parallel programs using an iterator-style interface //! //! You will rarely need to interact with this module directly unless you have //! need to name one of the iterator types. //! //! Parallel iterators make it easy to write iterator-like chains that //! execute in parallel: typically all you have to do is convert the //! first `.iter()` (or `iter_mut()`, `into_iter()`, etc) method into //! `par_iter()` (or `par_iter_mut()`, `into_par_iter()`, etc). For //! example, to compute the sum of the squares of a sequence of //! integers, one might write: //! //! ```rust //! use rayon::prelude::*; //! fn sum_of_squares(input: &[i32]) -> i32 { //! input.par_iter() //! .map(|i| i * i) //! .sum() //! } //! ``` //! //! Or, to increment all the integers in a slice, you could write: //! //! ```rust //! use rayon::prelude::*; //! fn increment_all(input: &mut [i32]) { //! input.par_iter_mut() //! .for_each(|p| *p += 1); //! } //! ``` //! //! To use parallel iterators, first import the traits by adding //! something like `use rayon::prelude::*` to your module. You can //! then call `par_iter`, `par_iter_mut`, or `into_par_iter` to get a //! parallel iterator. Like a [regular iterator][], parallel //! iterators work by first constructing a computation and then //! executing it. //! //! In addition to `par_iter()` and friends, some types offer other //! ways to create (or consume) parallel iterators: //! //! - Slices (`&[T]`, `&mut [T]`) offer methods like `par_split` and //! `par_windows`, as well as various parallel sorting //! operations. See [the `ParallelSlice` trait] for the full list. //! - Strings (`&str`) offer methods like `par_split` and `par_lines`. //! See [the `ParallelString` trait] for the full list. //! - Various collections offer [`par_extend`], which grows a //! collection given a parallel iterator. (If you don't have a //! collection to extend, you can use [`collect()`] to create a new //! one from scratch.) //! //! [the `ParallelSlice` trait]: ../slice/trait.ParallelSlice.html //! [the `ParallelString` trait]: ../str/trait.ParallelString.html //! [`par_extend`]: trait.ParallelExtend.html //! [`collect()`]: trait.ParallelIterator.html#method.collect //! //! To see the full range of methods available on parallel iterators, //! check out the [`ParallelIterator`] and [`IndexedParallelIterator`] //! traits. //! //! If you'd like to build a custom parallel iterator, or to write your own //! combinator, then check out the [split] function and the [plumbing] module. //! //! [regular iterator]: https://doc.rust-lang.org/std/iter/trait.Iterator.html //! [`ParallelIterator`]: trait.ParallelIterator.html //! [`IndexedParallelIterator`]: trait.IndexedParallelIterator.html //! [split]: fn.split.html //! [plumbing]: plumbing/index.html //! //! Note: Several of the `ParallelIterator` methods rely on a `Try` trait which //! has been deliberately obscured from the public API. This trait is intended //! to mirror the unstable `std::ops::Try` with implementations for `Option` and //! `Result`, where `Some`/`Ok` values will let those iterators continue, but //! `None`/`Err` values will exit early. //! //! A note about object safety: It is currently _not_ possible to wrap //! a `ParallelIterator` (or any trait that depends on it) using a //! `Box` or other kind of dynamic allocation, //! because `ParallelIterator` is **not object-safe**. //! (This keeps the implementation simpler and allows extra optimizations.) use self::plumbing::*; use self::private::Try; pub use either::Either; use std::cmp::{self, Ordering}; use std::iter::{Product, Sum}; use std::ops::{Fn, RangeBounds}; pub mod plumbing; #[cfg(test)] mod test; // There is a method to the madness here: // // - These modules are private but expose certain types to the end-user // (e.g., `enumerate::Enumerate`) -- specifically, the types that appear in the // public API surface of the `ParallelIterator` traits. // - In **this** module, those public types are always used unprefixed, which forces // us to add a `pub use` and helps identify if we missed anything. // - In contrast, items that appear **only** in the body of a method, // e.g. `find::find()`, are always used **prefixed**, so that they // can be readily distinguished. mod chain; mod chunks; mod cloned; mod collect; mod copied; mod empty; mod enumerate; mod extend; mod filter; mod filter_map; mod find; mod find_first_last; mod flat_map; mod flat_map_iter; mod flatten; mod flatten_iter; mod fold; mod fold_chunks; mod fold_chunks_with; mod for_each; mod from_par_iter; mod inspect; mod interleave; mod interleave_shortest; mod intersperse; mod len; mod map; mod map_with; mod multizip; mod noop; mod once; mod panic_fuse; mod par_bridge; mod positions; mod product; mod reduce; mod repeat; mod rev; mod skip; mod skip_any; mod skip_any_while; mod splitter; mod step_by; mod sum; mod take; mod take_any; mod take_any_while; mod try_fold; mod try_reduce; mod try_reduce_with; mod unzip; mod update; mod while_some; mod zip; mod zip_eq; pub use self::{ chain::Chain, chunks::Chunks, cloned::Cloned, copied::Copied, empty::{empty, Empty}, enumerate::Enumerate, filter::Filter, filter_map::FilterMap, flat_map::FlatMap, flat_map_iter::FlatMapIter, flatten::Flatten, flatten_iter::FlattenIter, fold::{Fold, FoldWith}, fold_chunks::FoldChunks, fold_chunks_with::FoldChunksWith, inspect::Inspect, interleave::Interleave, interleave_shortest::InterleaveShortest, intersperse::Intersperse, len::{MaxLen, MinLen}, map::Map, map_with::{MapInit, MapWith}, multizip::MultiZip, once::{once, Once}, panic_fuse::PanicFuse, par_bridge::{IterBridge, ParallelBridge}, positions::Positions, repeat::{repeat, repeatn, Repeat, RepeatN}, rev::Rev, skip::Skip, skip_any::SkipAny, skip_any_while::SkipAnyWhile, splitter::{split, Split}, step_by::StepBy, take::Take, take_any::TakeAny, take_any_while::TakeAnyWhile, try_fold::{TryFold, TryFoldWith}, update::Update, while_some::WhileSome, zip::Zip, zip_eq::ZipEq, }; /// `IntoParallelIterator` implements the conversion to a [`ParallelIterator`]. /// /// By implementing `IntoParallelIterator` for a type, you define how it will /// transformed into an iterator. This is a parallel version of the standard /// library's [`std::iter::IntoIterator`] trait. /// /// [`ParallelIterator`]: trait.ParallelIterator.html /// [`std::iter::IntoIterator`]: https://doc.rust-lang.org/std/iter/trait.IntoIterator.html pub trait IntoParallelIterator { /// The parallel iterator type that will be created. type Iter: ParallelIterator; /// The type of item that the parallel iterator will produce. type Item: Send; /// Converts `self` into a parallel iterator. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// println!("counting in parallel:"); /// (0..100).into_par_iter() /// .for_each(|i| println!("{}", i)); /// ``` /// /// This conversion is often implicit for arguments to methods like [`zip`]. /// /// ``` /// use rayon::prelude::*; /// /// let v: Vec<_> = (0..5).into_par_iter().zip(5..10).collect(); /// assert_eq!(v, [(0, 5), (1, 6), (2, 7), (3, 8), (4, 9)]); /// ``` /// /// [`zip`]: trait.IndexedParallelIterator.html#method.zip fn into_par_iter(self) -> Self::Iter; } /// `IntoParallelRefIterator` implements the conversion to a /// [`ParallelIterator`], providing shared references to the data. /// /// This is a parallel version of the `iter()` method /// defined by various collections. /// /// This trait is automatically implemented /// `for I where &I: IntoParallelIterator`. In most cases, users /// will want to implement [`IntoParallelIterator`] rather than implement /// this trait directly. /// /// [`ParallelIterator`]: trait.ParallelIterator.html /// [`IntoParallelIterator`]: trait.IntoParallelIterator.html pub trait IntoParallelRefIterator<'data> { /// The type of the parallel iterator that will be returned. type Iter: ParallelIterator; /// The type of item that the parallel iterator will produce. /// This will typically be an `&'data T` reference type. type Item: Send + 'data; /// Converts `self` into a parallel iterator. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let v: Vec<_> = (0..100).collect(); /// assert_eq!(v.par_iter().sum::(), 100 * 99 / 2); /// /// // `v.par_iter()` is shorthand for `(&v).into_par_iter()`, /// // producing the exact same references. /// assert!(v.par_iter().zip(&v) /// .all(|(a, b)| std::ptr::eq(a, b))); /// ``` fn par_iter(&'data self) -> Self::Iter; } impl<'data, I: 'data + ?Sized> IntoParallelRefIterator<'data> for I where &'data I: IntoParallelIterator, { type Iter = <&'data I as IntoParallelIterator>::Iter; type Item = <&'data I as IntoParallelIterator>::Item; fn par_iter(&'data self) -> Self::Iter { self.into_par_iter() } } /// `IntoParallelRefMutIterator` implements the conversion to a /// [`ParallelIterator`], providing mutable references to the data. /// /// This is a parallel version of the `iter_mut()` method /// defined by various collections. /// /// This trait is automatically implemented /// `for I where &mut I: IntoParallelIterator`. In most cases, users /// will want to implement [`IntoParallelIterator`] rather than implement /// this trait directly. /// /// [`ParallelIterator`]: trait.ParallelIterator.html /// [`IntoParallelIterator`]: trait.IntoParallelIterator.html pub trait IntoParallelRefMutIterator<'data> { /// The type of iterator that will be created. type Iter: ParallelIterator; /// The type of item that will be produced; this is typically an /// `&'data mut T` reference. type Item: Send + 'data; /// Creates the parallel iterator from `self`. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let mut v = vec![0usize; 5]; /// v.par_iter_mut().enumerate().for_each(|(i, x)| *x = i); /// assert_eq!(v, [0, 1, 2, 3, 4]); /// ``` fn par_iter_mut(&'data mut self) -> Self::Iter; } impl<'data, I: 'data + ?Sized> IntoParallelRefMutIterator<'data> for I where &'data mut I: IntoParallelIterator, { type Iter = <&'data mut I as IntoParallelIterator>::Iter; type Item = <&'data mut I as IntoParallelIterator>::Item; fn par_iter_mut(&'data mut self) -> Self::Iter { self.into_par_iter() } } /// Parallel version of the standard iterator trait. /// /// The combinators on this trait are available on **all** parallel /// iterators. Additional methods can be found on the /// [`IndexedParallelIterator`] trait: those methods are only /// available for parallel iterators where the number of items is /// known in advance (so, e.g., after invoking `filter`, those methods /// become unavailable). /// /// For examples of using parallel iterators, see [the docs on the /// `iter` module][iter]. /// /// [iter]: index.html /// [`IndexedParallelIterator`]: trait.IndexedParallelIterator.html pub trait ParallelIterator: Sized + Send { /// The type of item that this parallel iterator produces. /// For example, if you use the [`for_each`] method, this is the type of /// item that your closure will be invoked with. /// /// [`for_each`]: #method.for_each type Item: Send; /// Executes `OP` on each item produced by the iterator, in parallel. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// (0..100).into_par_iter().for_each(|x| println!("{:?}", x)); /// ``` fn for_each(self, op: OP) where OP: Fn(Self::Item) + Sync + Send, { for_each::for_each(self, &op) } /// Executes `OP` on the given `init` value with each item produced by /// the iterator, in parallel. /// /// The `init` value will be cloned only as needed to be paired with /// the group of items in each rayon job. It does not require the type /// to be `Sync`. /// /// # Examples /// /// ``` /// use std::sync::mpsc::channel; /// use rayon::prelude::*; /// /// let (sender, receiver) = channel(); /// /// (0..5).into_par_iter().for_each_with(sender, |s, x| s.send(x).unwrap()); /// /// let mut res: Vec<_> = receiver.iter().collect(); /// /// res.sort(); /// /// assert_eq!(&res[..], &[0, 1, 2, 3, 4]) /// ``` fn for_each_with(self, init: T, op: OP) where OP: Fn(&mut T, Self::Item) + Sync + Send, T: Send + Clone, { self.map_with(init, op).collect() } /// Executes `OP` on a value returned by `init` with each item produced by /// the iterator, in parallel. /// /// The `init` function will be called only as needed for a value to be /// paired with the group of items in each rayon job. There is no /// constraint on that returned type at all! /// /// # Examples /// /// ``` /// use rand::Rng; /// use rayon::prelude::*; /// /// let mut v = vec![0u8; 1_000_000]; /// /// v.par_chunks_mut(1000) /// .for_each_init( /// || rand::thread_rng(), /// |rng, chunk| rng.fill(chunk), /// ); /// /// // There's a remote chance that this will fail... /// for i in 0u8..=255 { /// assert!(v.contains(&i)); /// } /// ``` fn for_each_init(self, init: INIT, op: OP) where OP: Fn(&mut T, Self::Item) + Sync + Send, INIT: Fn() -> T + Sync + Send, { self.map_init(init, op).collect() } /// Executes a fallible `OP` on each item produced by the iterator, in parallel. /// /// If the `OP` returns `Result::Err` or `Option::None`, we will attempt to /// stop processing the rest of the items in the iterator as soon as /// possible, and we will return that terminating value. Otherwise, we will /// return an empty `Result::Ok(())` or `Option::Some(())`. If there are /// multiple errors in parallel, it is not specified which will be returned. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use std::io::{self, Write}; /// /// // This will stop iteration early if there's any write error, like /// // having piped output get closed on the other end. /// (0..100).into_par_iter() /// .try_for_each(|x| writeln!(io::stdout(), "{:?}", x)) /// .expect("expected no write errors"); /// ``` fn try_for_each(self, op: OP) -> R where OP: Fn(Self::Item) -> R + Sync + Send, R: Try + Send, { fn ok>(_: (), _: ()) -> R { R::from_output(()) } self.map(op).try_reduce(<()>::default, ok) } /// Executes a fallible `OP` on the given `init` value with each item /// produced by the iterator, in parallel. /// /// This combines the `init` semantics of [`for_each_with()`] and the /// failure semantics of [`try_for_each()`]. /// /// [`for_each_with()`]: #method.for_each_with /// [`try_for_each()`]: #method.try_for_each /// /// # Examples /// /// ``` /// use std::sync::mpsc::channel; /// use rayon::prelude::*; /// /// let (sender, receiver) = channel(); /// /// (0..5).into_par_iter() /// .try_for_each_with(sender, |s, x| s.send(x)) /// .expect("expected no send errors"); /// /// let mut res: Vec<_> = receiver.iter().collect(); /// /// res.sort(); /// /// assert_eq!(&res[..], &[0, 1, 2, 3, 4]) /// ``` fn try_for_each_with(self, init: T, op: OP) -> R where OP: Fn(&mut T, Self::Item) -> R + Sync + Send, T: Send + Clone, R: Try + Send, { fn ok>(_: (), _: ()) -> R { R::from_output(()) } self.map_with(init, op).try_reduce(<()>::default, ok) } /// Executes a fallible `OP` on a value returned by `init` with each item /// produced by the iterator, in parallel. /// /// This combines the `init` semantics of [`for_each_init()`] and the /// failure semantics of [`try_for_each()`]. /// /// [`for_each_init()`]: #method.for_each_init /// [`try_for_each()`]: #method.try_for_each /// /// # Examples /// /// ``` /// use rand::Rng; /// use rayon::prelude::*; /// /// let mut v = vec![0u8; 1_000_000]; /// /// v.par_chunks_mut(1000) /// .try_for_each_init( /// || rand::thread_rng(), /// |rng, chunk| rng.try_fill(chunk), /// ) /// .expect("expected no rand errors"); /// /// // There's a remote chance that this will fail... /// for i in 0u8..=255 { /// assert!(v.contains(&i)); /// } /// ``` fn try_for_each_init(self, init: INIT, op: OP) -> R where OP: Fn(&mut T, Self::Item) -> R + Sync + Send, INIT: Fn() -> T + Sync + Send, R: Try + Send, { fn ok>(_: (), _: ()) -> R { R::from_output(()) } self.map_init(init, op).try_reduce(<()>::default, ok) } /// Counts the number of items in this parallel iterator. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let count = (0..100).into_par_iter().count(); /// /// assert_eq!(count, 100); /// ``` fn count(self) -> usize { fn one(_: T) -> usize { 1 } self.map(one).sum() } /// Applies `map_op` to each item of this iterator, producing a new /// iterator with the results. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let mut par_iter = (0..5).into_par_iter().map(|x| x * 2); /// /// let doubles: Vec<_> = par_iter.collect(); /// /// assert_eq!(&doubles[..], &[0, 2, 4, 6, 8]); /// ``` fn map(self, map_op: F) -> Map where F: Fn(Self::Item) -> R + Sync + Send, R: Send, { Map::new(self, map_op) } /// Applies `map_op` to the given `init` value with each item of this /// iterator, producing a new iterator with the results. /// /// The `init` value will be cloned only as needed to be paired with /// the group of items in each rayon job. It does not require the type /// to be `Sync`. /// /// # Examples /// /// ``` /// use std::sync::mpsc::channel; /// use rayon::prelude::*; /// /// let (sender, receiver) = channel(); /// /// let a: Vec<_> = (0..5) /// .into_par_iter() // iterating over i32 /// .map_with(sender, |s, x| { /// s.send(x).unwrap(); // sending i32 values through the channel /// x // returning i32 /// }) /// .collect(); // collecting the returned values into a vector /// /// let mut b: Vec<_> = receiver.iter() // iterating over the values in the channel /// .collect(); // and collecting them /// b.sort(); /// /// assert_eq!(a, b); /// ``` fn map_with(self, init: T, map_op: F) -> MapWith where F: Fn(&mut T, Self::Item) -> R + Sync + Send, T: Send + Clone, R: Send, { MapWith::new(self, init, map_op) } /// Applies `map_op` to a value returned by `init` with each item of this /// iterator, producing a new iterator with the results. /// /// The `init` function will be called only as needed for a value to be /// paired with the group of items in each rayon job. There is no /// constraint on that returned type at all! /// /// # Examples /// /// ``` /// use rand::Rng; /// use rayon::prelude::*; /// /// let a: Vec<_> = (1i32..1_000_000) /// .into_par_iter() /// .map_init( /// || rand::thread_rng(), // get the thread-local RNG /// |rng, x| if rng.gen() { // randomly negate items /// -x /// } else { /// x /// }, /// ).collect(); /// /// // There's a remote chance that this will fail... /// assert!(a.iter().any(|&x| x < 0)); /// assert!(a.iter().any(|&x| x > 0)); /// ``` fn map_init(self, init: INIT, map_op: F) -> MapInit where F: Fn(&mut T, Self::Item) -> R + Sync + Send, INIT: Fn() -> T + Sync + Send, R: Send, { MapInit::new(self, init, map_op) } /// Creates an iterator which clones all of its elements. This may be /// useful when you have an iterator over `&T`, but you need `T`, and /// that type implements `Clone`. See also [`copied()`]. /// /// [`copied()`]: #method.copied /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3]; /// /// let v_cloned: Vec<_> = a.par_iter().cloned().collect(); /// /// // cloned is the same as .map(|&x| x), for integers /// let v_map: Vec<_> = a.par_iter().map(|&x| x).collect(); /// /// assert_eq!(v_cloned, vec![1, 2, 3]); /// assert_eq!(v_map, vec![1, 2, 3]); /// ``` fn cloned<'a, T>(self) -> Cloned where T: 'a + Clone + Send, Self: ParallelIterator, { Cloned::new(self) } /// Creates an iterator which copies all of its elements. This may be /// useful when you have an iterator over `&T`, but you need `T`, and /// that type implements `Copy`. See also [`cloned()`]. /// /// [`cloned()`]: #method.cloned /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3]; /// /// let v_copied: Vec<_> = a.par_iter().copied().collect(); /// /// // copied is the same as .map(|&x| x), for integers /// let v_map: Vec<_> = a.par_iter().map(|&x| x).collect(); /// /// assert_eq!(v_copied, vec![1, 2, 3]); /// assert_eq!(v_map, vec![1, 2, 3]); /// ``` fn copied<'a, T>(self) -> Copied where T: 'a + Copy + Send, Self: ParallelIterator, { Copied::new(self) } /// Applies `inspect_op` to a reference to each item of this iterator, /// producing a new iterator passing through the original items. This is /// often useful for debugging to see what's happening in iterator stages. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 4, 2, 3]; /// /// // this iterator sequence is complex. /// let sum = a.par_iter() /// .cloned() /// .filter(|&x| x % 2 == 0) /// .reduce(|| 0, |sum, i| sum + i); /// /// println!("{}", sum); /// /// // let's add some inspect() calls to investigate what's happening /// let sum = a.par_iter() /// .cloned() /// .inspect(|x| println!("about to filter: {}", x)) /// .filter(|&x| x % 2 == 0) /// .inspect(|x| println!("made it through filter: {}", x)) /// .reduce(|| 0, |sum, i| sum + i); /// /// println!("{}", sum); /// ``` fn inspect(self, inspect_op: OP) -> Inspect where OP: Fn(&Self::Item) + Sync + Send, { Inspect::new(self, inspect_op) } /// Mutates each item of this iterator before yielding it. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let par_iter = (0..5).into_par_iter().update(|x| {*x *= 2;}); /// /// let doubles: Vec<_> = par_iter.collect(); /// /// assert_eq!(&doubles[..], &[0, 2, 4, 6, 8]); /// ``` fn update(self, update_op: F) -> Update where F: Fn(&mut Self::Item) + Sync + Send, { Update::new(self, update_op) } /// Applies `filter_op` to each item of this iterator, producing a new /// iterator with only the items that gave `true` results. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let mut par_iter = (0..10).into_par_iter().filter(|x| x % 2 == 0); /// /// let even_numbers: Vec<_> = par_iter.collect(); /// /// assert_eq!(&even_numbers[..], &[0, 2, 4, 6, 8]); /// ``` fn filter

(self, filter_op: P) -> Filter where P: Fn(&Self::Item) -> bool + Sync + Send, { Filter::new(self, filter_op) } /// Applies `filter_op` to each item of this iterator to get an `Option`, /// producing a new iterator with only the items from `Some` results. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let mut par_iter = (0..10).into_par_iter() /// .filter_map(|x| { /// if x % 2 == 0 { Some(x * 3) } /// else { None } /// }); /// /// let even_numbers: Vec<_> = par_iter.collect(); /// /// assert_eq!(&even_numbers[..], &[0, 6, 12, 18, 24]); /// ``` fn filter_map(self, filter_op: P) -> FilterMap where P: Fn(Self::Item) -> Option + Sync + Send, R: Send, { FilterMap::new(self, filter_op) } /// Applies `map_op` to each item of this iterator to get nested parallel iterators, /// producing a new parallel iterator that flattens these back into one. /// /// See also [`flat_map_iter`](#method.flat_map_iter). /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [[1, 2], [3, 4], [5, 6], [7, 8]]; /// /// let par_iter = a.par_iter().cloned().flat_map(|a| a.to_vec()); /// /// let vec: Vec<_> = par_iter.collect(); /// /// assert_eq!(&vec[..], &[1, 2, 3, 4, 5, 6, 7, 8]); /// ``` fn flat_map(self, map_op: F) -> FlatMap where F: Fn(Self::Item) -> PI + Sync + Send, PI: IntoParallelIterator, { FlatMap::new(self, map_op) } /// Applies `map_op` to each item of this iterator to get nested serial iterators, /// producing a new parallel iterator that flattens these back into one. /// /// # `flat_map_iter` versus `flat_map` /// /// These two methods are similar but behave slightly differently. With [`flat_map`], /// each of the nested iterators must be a parallel iterator, and they will be further /// split up with nested parallelism. With `flat_map_iter`, each nested iterator is a /// sequential `Iterator`, and we only parallelize _between_ them, while the items /// produced by each nested iterator are processed sequentially. /// /// When choosing between these methods, consider whether nested parallelism suits the /// potential iterators at hand. If there's little computation involved, or its length /// is much less than the outer parallel iterator, then it may perform better to avoid /// the overhead of parallelism, just flattening sequentially with `flat_map_iter`. /// If there is a lot of computation, potentially outweighing the outer parallel /// iterator, then the nested parallelism of `flat_map` may be worthwhile. /// /// [`flat_map`]: #method.flat_map /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use std::cell::RefCell; /// /// let a = [[1, 2], [3, 4], [5, 6], [7, 8]]; /// /// let par_iter = a.par_iter().flat_map_iter(|a| { /// // The serial iterator doesn't have to be thread-safe, just its items. /// let cell_iter = RefCell::new(a.iter().cloned()); /// std::iter::from_fn(move || cell_iter.borrow_mut().next()) /// }); /// /// let vec: Vec<_> = par_iter.collect(); /// /// assert_eq!(&vec[..], &[1, 2, 3, 4, 5, 6, 7, 8]); /// ``` fn flat_map_iter(self, map_op: F) -> FlatMapIter where F: Fn(Self::Item) -> SI + Sync + Send, SI: IntoIterator, SI::Item: Send, { FlatMapIter::new(self, map_op) } /// An adaptor that flattens parallel-iterable `Item`s into one large iterator. /// /// See also [`flatten_iter`](#method.flatten_iter). /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let x: Vec> = vec![vec![1, 2], vec![3, 4]]; /// let y: Vec<_> = x.into_par_iter().flatten().collect(); /// /// assert_eq!(y, vec![1, 2, 3, 4]); /// ``` fn flatten(self) -> Flatten where Self::Item: IntoParallelIterator, { Flatten::new(self) } /// An adaptor that flattens serial-iterable `Item`s into one large iterator. /// /// See also [`flatten`](#method.flatten) and the analogous comparison of /// [`flat_map_iter` versus `flat_map`](#flat_map_iter-versus-flat_map). /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let x: Vec> = vec![vec![1, 2], vec![3, 4]]; /// let iters: Vec<_> = x.into_iter().map(Vec::into_iter).collect(); /// let y: Vec<_> = iters.into_par_iter().flatten_iter().collect(); /// /// assert_eq!(y, vec![1, 2, 3, 4]); /// ``` fn flatten_iter(self) -> FlattenIter where Self::Item: IntoIterator, ::Item: Send, { FlattenIter::new(self) } /// Reduces the items in the iterator into one item using `op`. /// The argument `identity` should be a closure that can produce /// "identity" value which may be inserted into the sequence as /// needed to create opportunities for parallel execution. So, for /// example, if you are doing a summation, then `identity()` ought /// to produce something that represents the zero for your type /// (but consider just calling `sum()` in that case). /// /// # Examples /// /// ``` /// // Iterate over a sequence of pairs `(x0, y0), ..., (xN, yN)` /// // and use reduce to compute one pair `(x0 + ... + xN, y0 + ... + yN)` /// // where the first/second elements are summed separately. /// use rayon::prelude::*; /// let sums = [(0, 1), (5, 6), (16, 2), (8, 9)] /// .par_iter() // iterating over &(i32, i32) /// .cloned() // iterating over (i32, i32) /// .reduce(|| (0, 0), // the "identity" is 0 in both columns /// |a, b| (a.0 + b.0, a.1 + b.1)); /// assert_eq!(sums, (0 + 5 + 16 + 8, 1 + 6 + 2 + 9)); /// ``` /// /// **Note:** unlike a sequential `fold` operation, the order in /// which `op` will be applied to reduce the result is not fully /// specified. So `op` should be [associative] or else the results /// will be non-deterministic. And of course `identity()` should /// produce a true identity. /// /// [associative]: https://en.wikipedia.org/wiki/Associative_property fn reduce(self, identity: ID, op: OP) -> Self::Item where OP: Fn(Self::Item, Self::Item) -> Self::Item + Sync + Send, ID: Fn() -> Self::Item + Sync + Send, { reduce::reduce(self, identity, op) } /// Reduces the items in the iterator into one item using `op`. /// If the iterator is empty, `None` is returned; otherwise, /// `Some` is returned. /// /// This version of `reduce` is simple but somewhat less /// efficient. If possible, it is better to call `reduce()`, which /// requires an identity element. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// let sums = [(0, 1), (5, 6), (16, 2), (8, 9)] /// .par_iter() // iterating over &(i32, i32) /// .cloned() // iterating over (i32, i32) /// .reduce_with(|a, b| (a.0 + b.0, a.1 + b.1)) /// .unwrap(); /// assert_eq!(sums, (0 + 5 + 16 + 8, 1 + 6 + 2 + 9)); /// ``` /// /// **Note:** unlike a sequential `fold` operation, the order in /// which `op` will be applied to reduce the result is not fully /// specified. So `op` should be [associative] or else the results /// will be non-deterministic. /// /// [associative]: https://en.wikipedia.org/wiki/Associative_property fn reduce_with(self, op: OP) -> Option where OP: Fn(Self::Item, Self::Item) -> Self::Item + Sync + Send, { fn opt_fold(op: impl Fn(T, T) -> T) -> impl Fn(Option, T) -> Option { move |opt_a, b| match opt_a { Some(a) => Some(op(a, b)), None => Some(b), } } fn opt_reduce(op: impl Fn(T, T) -> T) -> impl Fn(Option, Option) -> Option { move |opt_a, opt_b| match (opt_a, opt_b) { (Some(a), Some(b)) => Some(op(a, b)), (Some(v), None) | (None, Some(v)) => Some(v), (None, None) => None, } } self.fold(<_>::default, opt_fold(&op)) .reduce(<_>::default, opt_reduce(&op)) } /// Reduces the items in the iterator into one item using a fallible `op`. /// The `identity` argument is used the same way as in [`reduce()`]. /// /// [`reduce()`]: #method.reduce /// /// If a `Result::Err` or `Option::None` item is found, or if `op` reduces /// to one, we will attempt to stop processing the rest of the items in the /// iterator as soon as possible, and we will return that terminating value. /// Otherwise, we will return the final reduced `Result::Ok(T)` or /// `Option::Some(T)`. If there are multiple errors in parallel, it is not /// specified which will be returned. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// // Compute the sum of squares, being careful about overflow. /// fn sum_squares>(iter: I) -> Option { /// iter.into_par_iter() /// .map(|i| i.checked_mul(i)) // square each item, /// .try_reduce(|| 0, i32::checked_add) // and add them up! /// } /// assert_eq!(sum_squares(0..5), Some(0 + 1 + 4 + 9 + 16)); /// /// // The sum might overflow /// assert_eq!(sum_squares(0..10_000), None); /// /// // Or the squares might overflow before it even reaches `try_reduce` /// assert_eq!(sum_squares(1_000_000..1_000_001), None); /// ``` fn try_reduce(self, identity: ID, op: OP) -> Self::Item where OP: Fn(T, T) -> Self::Item + Sync + Send, ID: Fn() -> T + Sync + Send, Self::Item: Try, { try_reduce::try_reduce(self, identity, op) } /// Reduces the items in the iterator into one item using a fallible `op`. /// /// Like [`reduce_with()`], if the iterator is empty, `None` is returned; /// otherwise, `Some` is returned. Beyond that, it behaves like /// [`try_reduce()`] for handling `Err`/`None`. /// /// [`reduce_with()`]: #method.reduce_with /// [`try_reduce()`]: #method.try_reduce /// /// For instance, with `Option` items, the return value may be: /// - `None`, the iterator was empty /// - `Some(None)`, we stopped after encountering `None`. /// - `Some(Some(x))`, the entire iterator reduced to `x`. /// /// With `Result` items, the nesting is more obvious: /// - `None`, the iterator was empty /// - `Some(Err(e))`, we stopped after encountering an error `e`. /// - `Some(Ok(x))`, the entire iterator reduced to `x`. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let files = ["/dev/null", "/does/not/exist"]; /// /// // Find the biggest file /// files.into_par_iter() /// .map(|path| std::fs::metadata(path).map(|m| (path, m.len()))) /// .try_reduce_with(|a, b| { /// Ok(if a.1 >= b.1 { a } else { b }) /// }) /// .expect("Some value, since the iterator is not empty") /// .expect_err("not found"); /// ``` fn try_reduce_with(self, op: OP) -> Option where OP: Fn(T, T) -> Self::Item + Sync + Send, Self::Item: Try, { try_reduce_with::try_reduce_with(self, op) } /// Parallel fold is similar to sequential fold except that the /// sequence of items may be subdivided before it is /// folded. Consider a list of numbers like `22 3 77 89 46`. If /// you used sequential fold to add them (`fold(0, |a,b| a+b)`, /// you would wind up first adding 0 + 22, then 22 + 3, then 25 + /// 77, and so forth. The **parallel fold** works similarly except /// that it first breaks up your list into sublists, and hence /// instead of yielding up a single sum at the end, it yields up /// multiple sums. The number of results is nondeterministic, as /// is the point where the breaks occur. /// /// So if we did the same parallel fold (`fold(0, |a,b| a+b)`) on /// our example list, we might wind up with a sequence of two numbers, /// like so: /// /// ```notrust /// 22 3 77 89 46 /// | | /// 102 135 /// ``` /// /// Or perhaps these three numbers: /// /// ```notrust /// 22 3 77 89 46 /// | | | /// 102 89 46 /// ``` /// /// In general, Rayon will attempt to find good breaking points /// that keep all of your cores busy. /// /// ### Fold versus reduce /// /// The `fold()` and `reduce()` methods each take an identity element /// and a combining function, but they operate rather differently. /// /// `reduce()` requires that the identity function has the same /// type as the things you are iterating over, and it fully /// reduces the list of items into a single item. So, for example, /// imagine we are iterating over a list of bytes `bytes: [128_u8, /// 64_u8, 64_u8]`. If we used `bytes.reduce(|| 0_u8, |a: u8, b: /// u8| a + b)`, we would get an overflow. This is because `0`, /// `a`, and `b` here are all bytes, just like the numbers in the /// list (I wrote the types explicitly above, but those are the /// only types you can use). To avoid the overflow, we would need /// to do something like `bytes.map(|b| b as u32).reduce(|| 0, |a, /// b| a + b)`, in which case our result would be `256`. /// /// In contrast, with `fold()`, the identity function does not /// have to have the same type as the things you are iterating /// over, and you potentially get back many results. So, if we /// continue with the `bytes` example from the previous paragraph, /// we could do `bytes.fold(|| 0_u32, |a, b| a + (b as u32))` to /// convert our bytes into `u32`. And of course we might not get /// back a single sum. /// /// There is a more subtle distinction as well, though it's /// actually implied by the above points. When you use `reduce()`, /// your reduction function is sometimes called with values that /// were never part of your original parallel iterator (for /// example, both the left and right might be a partial sum). With /// `fold()`, in contrast, the left value in the fold function is /// always the accumulator, and the right value is always from /// your original sequence. /// /// ### Fold vs Map/Reduce /// /// Fold makes sense if you have some operation where it is /// cheaper to create groups of elements at a time. For example, /// imagine collecting characters into a string. If you were going /// to use map/reduce, you might try this: /// /// ``` /// use rayon::prelude::*; /// /// let s = /// ['a', 'b', 'c', 'd', 'e'] /// .par_iter() /// .map(|c: &char| format!("{}", c)) /// .reduce(|| String::new(), /// |mut a: String, b: String| { a.push_str(&b); a }); /// /// assert_eq!(s, "abcde"); /// ``` /// /// Because reduce produces the same type of element as its input, /// you have to first map each character into a string, and then /// you can reduce them. This means we create one string per /// element in our iterator -- not so great. Using `fold`, we can /// do this instead: /// /// ``` /// use rayon::prelude::*; /// /// let s = /// ['a', 'b', 'c', 'd', 'e'] /// .par_iter() /// .fold(|| String::new(), /// |mut s: String, c: &char| { s.push(*c); s }) /// .reduce(|| String::new(), /// |mut a: String, b: String| { a.push_str(&b); a }); /// /// assert_eq!(s, "abcde"); /// ``` /// /// Now `fold` will process groups of our characters at a time, /// and we only make one string per group. We should wind up with /// some small-ish number of strings roughly proportional to the /// number of CPUs you have (it will ultimately depend on how busy /// your processors are). Note that we still need to do a reduce /// afterwards to combine those groups of strings into a single /// string. /// /// You could use a similar trick to save partial results (e.g., a /// cache) or something similar. /// /// ### Combining fold with other operations /// /// You can combine `fold` with `reduce` if you want to produce a /// single value. This is then roughly equivalent to a map/reduce /// combination in effect: /// /// ``` /// use rayon::prelude::*; /// /// let bytes = 0..22_u8; /// let sum = bytes.into_par_iter() /// .fold(|| 0_u32, |a: u32, b: u8| a + (b as u32)) /// .sum::(); /// /// assert_eq!(sum, (0..22).sum()); // compare to sequential /// ``` fn fold(self, identity: ID, fold_op: F) -> Fold where F: Fn(T, Self::Item) -> T + Sync + Send, ID: Fn() -> T + Sync + Send, T: Send, { Fold::new(self, identity, fold_op) } /// Applies `fold_op` to the given `init` value with each item of this /// iterator, finally producing the value for further use. /// /// This works essentially like `fold(|| init.clone(), fold_op)`, except /// it doesn't require the `init` type to be `Sync`, nor any other form /// of added synchronization. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let bytes = 0..22_u8; /// let sum = bytes.into_par_iter() /// .fold_with(0_u32, |a: u32, b: u8| a + (b as u32)) /// .sum::(); /// /// assert_eq!(sum, (0..22).sum()); // compare to sequential /// ``` fn fold_with(self, init: T, fold_op: F) -> FoldWith where F: Fn(T, Self::Item) -> T + Sync + Send, T: Send + Clone, { FoldWith::new(self, init, fold_op) } /// Performs a fallible parallel fold. /// /// This is a variation of [`fold()`] for operations which can fail with /// `Option::None` or `Result::Err`. The first such failure stops /// processing the local set of items, without affecting other folds in the /// iterator's subdivisions. /// /// Often, `try_fold()` will be followed by [`try_reduce()`] /// for a final reduction and global short-circuiting effect. /// /// [`fold()`]: #method.fold /// [`try_reduce()`]: #method.try_reduce /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let bytes = 0..22_u8; /// let sum = bytes.into_par_iter() /// .try_fold(|| 0_u32, |a: u32, b: u8| a.checked_add(b as u32)) /// .try_reduce(|| 0, u32::checked_add); /// /// assert_eq!(sum, Some((0..22).sum())); // compare to sequential /// ``` fn try_fold(self, identity: ID, fold_op: F) -> TryFold where F: Fn(T, Self::Item) -> R + Sync + Send, ID: Fn() -> T + Sync + Send, R: Try + Send, { TryFold::new(self, identity, fold_op) } /// Performs a fallible parallel fold with a cloneable `init` value. /// /// This combines the `init` semantics of [`fold_with()`] and the failure /// semantics of [`try_fold()`]. /// /// [`fold_with()`]: #method.fold_with /// [`try_fold()`]: #method.try_fold /// /// ``` /// use rayon::prelude::*; /// /// let bytes = 0..22_u8; /// let sum = bytes.into_par_iter() /// .try_fold_with(0_u32, |a: u32, b: u8| a.checked_add(b as u32)) /// .try_reduce(|| 0, u32::checked_add); /// /// assert_eq!(sum, Some((0..22).sum())); // compare to sequential /// ``` fn try_fold_with(self, init: T, fold_op: F) -> TryFoldWith where F: Fn(T, Self::Item) -> R + Sync + Send, R: Try + Send, T: Clone + Send, { TryFoldWith::new(self, init, fold_op) } /// Sums up the items in the iterator. /// /// Note that the order in items will be reduced is not specified, /// so if the `+` operator is not truly [associative] \(as is the /// case for floating point numbers), then the results are not /// fully deterministic. /// /// [associative]: https://en.wikipedia.org/wiki/Associative_property /// /// Basically equivalent to `self.reduce(|| 0, |a, b| a + b)`, /// except that the type of `0` and the `+` operation may vary /// depending on the type of value being produced. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 5, 7]; /// /// let sum: i32 = a.par_iter().sum(); /// /// assert_eq!(sum, 13); /// ``` fn sum(self) -> S where S: Send + Sum + Sum, { sum::sum(self) } /// Multiplies all the items in the iterator. /// /// Note that the order in items will be reduced is not specified, /// so if the `*` operator is not truly [associative] \(as is the /// case for floating point numbers), then the results are not /// fully deterministic. /// /// [associative]: https://en.wikipedia.org/wiki/Associative_property /// /// Basically equivalent to `self.reduce(|| 1, |a, b| a * b)`, /// except that the type of `1` and the `*` operation may vary /// depending on the type of value being produced. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// fn factorial(n: u32) -> u32 { /// (1..n+1).into_par_iter().product() /// } /// /// assert_eq!(factorial(0), 1); /// assert_eq!(factorial(1), 1); /// assert_eq!(factorial(5), 120); /// ``` fn product

(self) -> P where P: Send + Product + Product

, { product::product(self) } /// Computes the minimum of all the items in the iterator. If the /// iterator is empty, `None` is returned; otherwise, `Some(min)` /// is returned. /// /// Note that the order in which the items will be reduced is not /// specified, so if the `Ord` impl is not truly associative, then /// the results are not deterministic. /// /// Basically equivalent to `self.reduce_with(|a, b| cmp::min(a, b))`. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [45, 74, 32]; /// /// assert_eq!(a.par_iter().min(), Some(&32)); /// /// let b: [i32; 0] = []; /// /// assert_eq!(b.par_iter().min(), None); /// ``` fn min(self) -> Option where Self::Item: Ord, { self.reduce_with(cmp::min) } /// Computes the minimum of all the items in the iterator with respect to /// the given comparison function. If the iterator is empty, `None` is /// returned; otherwise, `Some(min)` is returned. /// /// Note that the order in which the items will be reduced is not /// specified, so if the comparison function is not associative, then /// the results are not deterministic. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [-3_i32, 77, 53, 240, -1]; /// /// assert_eq!(a.par_iter().min_by(|x, y| x.cmp(y)), Some(&-3)); /// ``` fn min_by(self, f: F) -> Option where F: Sync + Send + Fn(&Self::Item, &Self::Item) -> Ordering, { fn min(f: impl Fn(&T, &T) -> Ordering) -> impl Fn(T, T) -> T { move |a, b| match f(&a, &b) { Ordering::Greater => b, _ => a, } } self.reduce_with(min(f)) } /// Computes the item that yields the minimum value for the given /// function. If the iterator is empty, `None` is returned; /// otherwise, `Some(item)` is returned. /// /// Note that the order in which the items will be reduced is not /// specified, so if the `Ord` impl is not truly associative, then /// the results are not deterministic. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [-3_i32, 34, 2, 5, -10, -3, -23]; /// /// assert_eq!(a.par_iter().min_by_key(|x| x.abs()), Some(&2)); /// ``` fn min_by_key(self, f: F) -> Option where K: Ord + Send, F: Sync + Send + Fn(&Self::Item) -> K, { fn key(f: impl Fn(&T) -> K) -> impl Fn(T) -> (K, T) { move |x| (f(&x), x) } fn min_key(a: (K, T), b: (K, T)) -> (K, T) { match (a.0).cmp(&b.0) { Ordering::Greater => b, _ => a, } } let (_, x) = self.map(key(f)).reduce_with(min_key)?; Some(x) } /// Computes the maximum of all the items in the iterator. If the /// iterator is empty, `None` is returned; otherwise, `Some(max)` /// is returned. /// /// Note that the order in which the items will be reduced is not /// specified, so if the `Ord` impl is not truly associative, then /// the results are not deterministic. /// /// Basically equivalent to `self.reduce_with(|a, b| cmp::max(a, b))`. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [45, 74, 32]; /// /// assert_eq!(a.par_iter().max(), Some(&74)); /// /// let b: [i32; 0] = []; /// /// assert_eq!(b.par_iter().max(), None); /// ``` fn max(self) -> Option where Self::Item: Ord, { self.reduce_with(cmp::max) } /// Computes the maximum of all the items in the iterator with respect to /// the given comparison function. If the iterator is empty, `None` is /// returned; otherwise, `Some(max)` is returned. /// /// Note that the order in which the items will be reduced is not /// specified, so if the comparison function is not associative, then /// the results are not deterministic. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [-3_i32, 77, 53, 240, -1]; /// /// assert_eq!(a.par_iter().max_by(|x, y| x.abs().cmp(&y.abs())), Some(&240)); /// ``` fn max_by(self, f: F) -> Option where F: Sync + Send + Fn(&Self::Item, &Self::Item) -> Ordering, { fn max(f: impl Fn(&T, &T) -> Ordering) -> impl Fn(T, T) -> T { move |a, b| match f(&a, &b) { Ordering::Greater => a, _ => b, } } self.reduce_with(max(f)) } /// Computes the item that yields the maximum value for the given /// function. If the iterator is empty, `None` is returned; /// otherwise, `Some(item)` is returned. /// /// Note that the order in which the items will be reduced is not /// specified, so if the `Ord` impl is not truly associative, then /// the results are not deterministic. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [-3_i32, 34, 2, 5, -10, -3, -23]; /// /// assert_eq!(a.par_iter().max_by_key(|x| x.abs()), Some(&34)); /// ``` fn max_by_key(self, f: F) -> Option where K: Ord + Send, F: Sync + Send + Fn(&Self::Item) -> K, { fn key(f: impl Fn(&T) -> K) -> impl Fn(T) -> (K, T) { move |x| (f(&x), x) } fn max_key(a: (K, T), b: (K, T)) -> (K, T) { match (a.0).cmp(&b.0) { Ordering::Greater => a, _ => b, } } let (_, x) = self.map(key(f)).reduce_with(max_key)?; Some(x) } /// Takes two iterators and creates a new iterator over both. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [0, 1, 2]; /// let b = [9, 8, 7]; /// /// let par_iter = a.par_iter().chain(b.par_iter()); /// /// let chained: Vec<_> = par_iter.cloned().collect(); /// /// assert_eq!(&chained[..], &[0, 1, 2, 9, 8, 7]); /// ``` fn chain(self, chain: C) -> Chain where C: IntoParallelIterator, { Chain::new(self, chain.into_par_iter()) } /// Searches for **some** item in the parallel iterator that /// matches the given predicate and returns it. This operation /// is similar to [`find` on sequential iterators][find] but /// the item returned may not be the **first** one in the parallel /// sequence which matches, since we search the entire sequence in parallel. /// /// Once a match is found, we will attempt to stop processing /// the rest of the items in the iterator as soon as possible /// (just as `find` stops iterating once a match is found). /// /// [find]: https://doc.rust-lang.org/std/iter/trait.Iterator.html#method.find /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3, 3]; /// /// assert_eq!(a.par_iter().find_any(|&&x| x == 3), Some(&3)); /// /// assert_eq!(a.par_iter().find_any(|&&x| x == 100), None); /// ``` fn find_any

(self, predicate: P) -> Option where P: Fn(&Self::Item) -> bool + Sync + Send, { find::find(self, predicate) } /// Searches for the sequentially **first** item in the parallel iterator /// that matches the given predicate and returns it. /// /// Once a match is found, all attempts to the right of the match /// will be stopped, while attempts to the left must continue in case /// an earlier match is found. /// /// Note that not all parallel iterators have a useful order, much like /// sequential `HashMap` iteration, so "first" may be nebulous. If you /// just want the first match that discovered anywhere in the iterator, /// `find_any` is a better choice. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3, 3]; /// /// assert_eq!(a.par_iter().find_first(|&&x| x == 3), Some(&3)); /// /// assert_eq!(a.par_iter().find_first(|&&x| x == 100), None); /// ``` fn find_first

(self, predicate: P) -> Option where P: Fn(&Self::Item) -> bool + Sync + Send, { find_first_last::find_first(self, predicate) } /// Searches for the sequentially **last** item in the parallel iterator /// that matches the given predicate and returns it. /// /// Once a match is found, all attempts to the left of the match /// will be stopped, while attempts to the right must continue in case /// a later match is found. /// /// Note that not all parallel iterators have a useful order, much like /// sequential `HashMap` iteration, so "last" may be nebulous. When the /// order doesn't actually matter to you, `find_any` is a better choice. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3, 3]; /// /// assert_eq!(a.par_iter().find_last(|&&x| x == 3), Some(&3)); /// /// assert_eq!(a.par_iter().find_last(|&&x| x == 100), None); /// ``` fn find_last

(self, predicate: P) -> Option where P: Fn(&Self::Item) -> bool + Sync + Send, { find_first_last::find_last(self, predicate) } /// Applies the given predicate to the items in the parallel iterator /// and returns **any** non-None result of the map operation. /// /// Once a non-None value is produced from the map operation, we will /// attempt to stop processing the rest of the items in the iterator /// as soon as possible. /// /// Note that this method only returns **some** item in the parallel /// iterator that is not None from the map predicate. The item returned /// may not be the **first** non-None value produced in the parallel /// sequence, since the entire sequence is mapped over in parallel. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let c = ["lol", "NaN", "5", "5"]; /// /// let found_number = c.par_iter().find_map_any(|s| s.parse().ok()); /// /// assert_eq!(found_number, Some(5)); /// ``` fn find_map_any(self, predicate: P) -> Option where P: Fn(Self::Item) -> Option + Sync + Send, R: Send, { fn yes(_: &T) -> bool { true } self.filter_map(predicate).find_any(yes) } /// Applies the given predicate to the items in the parallel iterator and /// returns the sequentially **first** non-None result of the map operation. /// /// Once a non-None value is produced from the map operation, all attempts /// to the right of the match will be stopped, while attempts to the left /// must continue in case an earlier match is found. /// /// Note that not all parallel iterators have a useful order, much like /// sequential `HashMap` iteration, so "first" may be nebulous. If you /// just want the first non-None value discovered anywhere in the iterator, /// `find_map_any` is a better choice. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let c = ["lol", "NaN", "2", "5"]; /// /// let first_number = c.par_iter().find_map_first(|s| s.parse().ok()); /// /// assert_eq!(first_number, Some(2)); /// ``` fn find_map_first(self, predicate: P) -> Option where P: Fn(Self::Item) -> Option + Sync + Send, R: Send, { fn yes(_: &T) -> bool { true } self.filter_map(predicate).find_first(yes) } /// Applies the given predicate to the items in the parallel iterator and /// returns the sequentially **last** non-None result of the map operation. /// /// Once a non-None value is produced from the map operation, all attempts /// to the left of the match will be stopped, while attempts to the right /// must continue in case a later match is found. /// /// Note that not all parallel iterators have a useful order, much like /// sequential `HashMap` iteration, so "first" may be nebulous. If you /// just want the first non-None value discovered anywhere in the iterator, /// `find_map_any` is a better choice. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let c = ["lol", "NaN", "2", "5"]; /// /// let last_number = c.par_iter().find_map_last(|s| s.parse().ok()); /// /// assert_eq!(last_number, Some(5)); /// ``` fn find_map_last(self, predicate: P) -> Option where P: Fn(Self::Item) -> Option + Sync + Send, R: Send, { fn yes(_: &T) -> bool { true } self.filter_map(predicate).find_last(yes) } #[doc(hidden)] #[deprecated(note = "parallel `find` does not search in order -- use `find_any`, \\ `find_first`, or `find_last`")] fn find

(self, predicate: P) -> Option where P: Fn(&Self::Item) -> bool + Sync + Send, { self.find_any(predicate) } /// Searches for **some** item in the parallel iterator that /// matches the given predicate, and if so returns true. Once /// a match is found, we'll attempt to stop process the rest /// of the items. Proving that there's no match, returning false, /// does require visiting every item. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [0, 12, 3, 4, 0, 23, 0]; /// /// let is_valid = a.par_iter().any(|&x| x > 10); /// /// assert!(is_valid); /// ``` fn any

(self, predicate: P) -> bool where P: Fn(Self::Item) -> bool + Sync + Send, { self.map(predicate).find_any(bool::clone).is_some() } /// Tests that every item in the parallel iterator matches the given /// predicate, and if so returns true. If a counter-example is found, /// we'll attempt to stop processing more items, then return false. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [0, 12, 3, 4, 0, 23, 0]; /// /// let is_valid = a.par_iter().all(|&x| x > 10); /// /// assert!(!is_valid); /// ``` fn all

(self, predicate: P) -> bool where P: Fn(Self::Item) -> bool + Sync + Send, { #[inline] fn is_false(x: &bool) -> bool { !x } self.map(predicate).find_any(is_false).is_none() } /// Creates an iterator over the `Some` items of this iterator, halting /// as soon as any `None` is found. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use std::sync::atomic::{AtomicUsize, Ordering}; /// /// let counter = AtomicUsize::new(0); /// let value = (0_i32..2048) /// .into_par_iter() /// .map(|x| { /// counter.fetch_add(1, Ordering::SeqCst); /// if x < 1024 { Some(x) } else { None } /// }) /// .while_some() /// .max(); /// /// assert!(value < Some(1024)); /// assert!(counter.load(Ordering::SeqCst) < 2048); // should not have visited every single one /// ``` fn while_some(self) -> WhileSome where Self: ParallelIterator>, T: Send, { WhileSome::new(self) } /// Wraps an iterator with a fuse in case of panics, to halt all threads /// as soon as possible. /// /// Panics within parallel iterators are always propagated to the caller, /// but they don't always halt the rest of the iterator right away, due to /// the internal semantics of [`join`]. This adaptor makes a greater effort /// to stop processing other items sooner, with the cost of additional /// synchronization overhead, which may also inhibit some optimizations. /// /// [`join`]: ../fn.join.html#panics /// /// # Examples /// /// If this code didn't use `panic_fuse()`, it would continue processing /// many more items in other threads (with long sleep delays) before the /// panic is finally propagated. /// /// ```should_panic /// use rayon::prelude::*; /// use std::{thread, time}; /// /// (0..1_000_000) /// .into_par_iter() /// .panic_fuse() /// .for_each(|i| { /// // simulate some work /// thread::sleep(time::Duration::from_secs(1)); /// assert!(i > 0); // oops! /// }); /// ``` fn panic_fuse(self) -> PanicFuse { PanicFuse::new(self) } /// Creates a fresh collection containing all the elements produced /// by this parallel iterator. /// /// You may prefer [`collect_into_vec()`] implemented on /// [`IndexedParallelIterator`], if your underlying iterator also implements /// it. [`collect_into_vec()`] allocates efficiently with precise knowledge /// of how many elements the iterator contains, and even allows you to reuse /// an existing vector's backing store rather than allocating a fresh vector. /// /// [`IndexedParallelIterator`]: trait.IndexedParallelIterator.html /// [`collect_into_vec()`]: /// trait.IndexedParallelIterator.html#method.collect_into_vec /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let sync_vec: Vec<_> = (0..100).into_iter().collect(); /// /// let async_vec: Vec<_> = (0..100).into_par_iter().collect(); /// /// assert_eq!(sync_vec, async_vec); /// ``` /// /// You can collect a pair of collections like [`unzip`](#method.unzip) /// for paired items: /// /// ``` /// use rayon::prelude::*; /// /// let a = [(0, 1), (1, 2), (2, 3), (3, 4)]; /// let (first, second): (Vec<_>, Vec<_>) = a.into_par_iter().collect(); /// /// assert_eq!(first, [0, 1, 2, 3]); /// assert_eq!(second, [1, 2, 3, 4]); /// ``` /// /// Or like [`partition_map`](#method.partition_map) for `Either` items: /// /// ``` /// use rayon::prelude::*; /// use rayon::iter::Either; /// /// let (left, right): (Vec<_>, Vec<_>) = (0..8).into_par_iter().map(|x| { /// if x % 2 == 0 { /// Either::Left(x * 4) /// } else { /// Either::Right(x * 3) /// } /// }).collect(); /// /// assert_eq!(left, [0, 8, 16, 24]); /// assert_eq!(right, [3, 9, 15, 21]); /// ``` /// /// You can even collect an arbitrarily-nested combination of pairs and `Either`: /// /// ``` /// use rayon::prelude::*; /// use rayon::iter::Either; /// /// let (first, (left, right)): (Vec<_>, (Vec<_>, Vec<_>)) /// = (0..8).into_par_iter().map(|x| { /// if x % 2 == 0 { /// (x, Either::Left(x * 4)) /// } else { /// (-x, Either::Right(x * 3)) /// } /// }).collect(); /// /// assert_eq!(first, [0, -1, 2, -3, 4, -5, 6, -7]); /// assert_eq!(left, [0, 8, 16, 24]); /// assert_eq!(right, [3, 9, 15, 21]); /// ``` /// /// All of that can _also_ be combined with short-circuiting collection of /// `Result` or `Option` types: /// /// ``` /// use rayon::prelude::*; /// use rayon::iter::Either; /// /// let result: Result<(Vec<_>, (Vec<_>, Vec<_>)), _> /// = (0..8).into_par_iter().map(|x| { /// if x > 5 { /// Err(x) /// } else if x % 2 == 0 { /// Ok((x, Either::Left(x * 4))) /// } else { /// Ok((-x, Either::Right(x * 3))) /// } /// }).collect(); /// /// let error = result.unwrap_err(); /// assert!(error == 6 || error == 7); /// ``` fn collect(self) -> C where C: FromParallelIterator, { C::from_par_iter(self) } /// Unzips the items of a parallel iterator into a pair of arbitrary /// `ParallelExtend` containers. /// /// You may prefer to use `unzip_into_vecs()`, which allocates more /// efficiently with precise knowledge of how many elements the /// iterator contains, and even allows you to reuse existing /// vectors' backing stores rather than allocating fresh vectors. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [(0, 1), (1, 2), (2, 3), (3, 4)]; /// /// let (left, right): (Vec<_>, Vec<_>) = a.par_iter().cloned().unzip(); /// /// assert_eq!(left, [0, 1, 2, 3]); /// assert_eq!(right, [1, 2, 3, 4]); /// ``` /// /// Nested pairs can be unzipped too. /// /// ``` /// use rayon::prelude::*; /// /// let (values, (squares, cubes)): (Vec<_>, (Vec<_>, Vec<_>)) = (0..4).into_par_iter() /// .map(|i| (i, (i * i, i * i * i))) /// .unzip(); /// /// assert_eq!(values, [0, 1, 2, 3]); /// assert_eq!(squares, [0, 1, 4, 9]); /// assert_eq!(cubes, [0, 1, 8, 27]); /// ``` fn unzip(self) -> (FromA, FromB) where Self: ParallelIterator, FromA: Default + Send + ParallelExtend, FromB: Default + Send + ParallelExtend, A: Send, B: Send, { unzip::unzip(self) } /// Partitions the items of a parallel iterator into a pair of arbitrary /// `ParallelExtend` containers. Items for which the `predicate` returns /// true go into the first container, and the rest go into the second. /// /// Note: unlike the standard `Iterator::partition`, this allows distinct /// collection types for the left and right items. This is more flexible, /// but may require new type annotations when converting sequential code /// that used type inference assuming the two were the same. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let (left, right): (Vec<_>, Vec<_>) = (0..8).into_par_iter().partition(|x| x % 2 == 0); /// /// assert_eq!(left, [0, 2, 4, 6]); /// assert_eq!(right, [1, 3, 5, 7]); /// ``` fn partition(self, predicate: P) -> (A, B) where A: Default + Send + ParallelExtend, B: Default + Send + ParallelExtend, P: Fn(&Self::Item) -> bool + Sync + Send, { unzip::partition(self, predicate) } /// Partitions and maps the items of a parallel iterator into a pair of /// arbitrary `ParallelExtend` containers. `Either::Left` items go into /// the first container, and `Either::Right` items go into the second. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use rayon::iter::Either; /// /// let (left, right): (Vec<_>, Vec<_>) = (0..8).into_par_iter() /// .partition_map(|x| { /// if x % 2 == 0 { /// Either::Left(x * 4) /// } else { /// Either::Right(x * 3) /// } /// }); /// /// assert_eq!(left, [0, 8, 16, 24]); /// assert_eq!(right, [3, 9, 15, 21]); /// ``` /// /// Nested `Either` enums can be split as well. /// /// ``` /// use rayon::prelude::*; /// use rayon::iter::Either::*; /// /// let ((fizzbuzz, fizz), (buzz, other)): ((Vec<_>, Vec<_>), (Vec<_>, Vec<_>)) = (1..20) /// .into_par_iter() /// .partition_map(|x| match (x % 3, x % 5) { /// (0, 0) => Left(Left(x)), /// (0, _) => Left(Right(x)), /// (_, 0) => Right(Left(x)), /// (_, _) => Right(Right(x)), /// }); /// /// assert_eq!(fizzbuzz, [15]); /// assert_eq!(fizz, [3, 6, 9, 12, 18]); /// assert_eq!(buzz, [5, 10]); /// assert_eq!(other, [1, 2, 4, 7, 8, 11, 13, 14, 16, 17, 19]); /// ``` fn partition_map(self, predicate: P) -> (A, B) where A: Default + Send + ParallelExtend, B: Default + Send + ParallelExtend, P: Fn(Self::Item) -> Either + Sync + Send, L: Send, R: Send, { unzip::partition_map(self, predicate) } /// Intersperses clones of an element between items of this iterator. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let x = vec![1, 2, 3]; /// let r: Vec<_> = x.into_par_iter().intersperse(-1).collect(); /// /// assert_eq!(r, vec![1, -1, 2, -1, 3]); /// ``` fn intersperse(self, element: Self::Item) -> Intersperse where Self::Item: Clone, { Intersperse::new(self, element) } /// Creates an iterator that yields `n` elements from *anywhere* in the original iterator. /// /// This is similar to [`IndexedParallelIterator::take`] without being /// constrained to the "first" `n` of the original iterator order. The /// taken items will still maintain their relative order where that is /// visible in `collect`, `reduce`, and similar outputs. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..100) /// .into_par_iter() /// .filter(|&x| x % 2 == 0) /// .take_any(5) /// .collect(); /// /// assert_eq!(result.len(), 5); /// assert!(result.windows(2).all(|w| w[0] < w[1])); /// ``` fn take_any(self, n: usize) -> TakeAny { TakeAny::new(self, n) } /// Creates an iterator that skips `n` elements from *anywhere* in the original iterator. /// /// This is similar to [`IndexedParallelIterator::skip`] without being /// constrained to the "first" `n` of the original iterator order. The /// remaining items will still maintain their relative order where that is /// visible in `collect`, `reduce`, and similar outputs. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..100) /// .into_par_iter() /// .filter(|&x| x % 2 == 0) /// .skip_any(5) /// .collect(); /// /// assert_eq!(result.len(), 45); /// assert!(result.windows(2).all(|w| w[0] < w[1])); /// ``` fn skip_any(self, n: usize) -> SkipAny { SkipAny::new(self, n) } /// Creates an iterator that takes elements from *anywhere* in the original iterator /// until the given `predicate` returns `false`. /// /// The `predicate` may be anything -- e.g. it could be checking a fact about the item, a /// global condition unrelated to the item itself, or some combination thereof. /// /// If parallel calls to the `predicate` race and give different results, then the /// `true` results will still take those particular items, while respecting the `false` /// result from elsewhere to skip any further items. /// /// This is similar to [`Iterator::take_while`] without being constrained to the original /// iterator order. The taken items will still maintain their relative order where that is /// visible in `collect`, `reduce`, and similar outputs. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..100) /// .into_par_iter() /// .take_any_while(|x| *x < 50) /// .collect(); /// /// assert!(result.len() <= 50); /// assert!(result.windows(2).all(|w| w[0] < w[1])); /// ``` /// /// ``` /// use rayon::prelude::*; /// use std::sync::atomic::AtomicUsize; /// use std::sync::atomic::Ordering::Relaxed; /// /// // Collect any group of items that sum <= 1000 /// let quota = AtomicUsize::new(1000); /// let result: Vec<_> = (0_usize..100) /// .into_par_iter() /// .take_any_while(|&x| { /// quota.fetch_update(Relaxed, Relaxed, |q| q.checked_sub(x)) /// .is_ok() /// }) /// .collect(); /// /// let sum = result.iter().sum::(); /// assert!(matches!(sum, 902..=1000)); /// ``` fn take_any_while

(self, predicate: P) -> TakeAnyWhile where P: Fn(&Self::Item) -> bool + Sync + Send, { TakeAnyWhile::new(self, predicate) } /// Creates an iterator that skips elements from *anywhere* in the original iterator /// until the given `predicate` returns `false`. /// /// The `predicate` may be anything -- e.g. it could be checking a fact about the item, a /// global condition unrelated to the item itself, or some combination thereof. /// /// If parallel calls to the `predicate` race and give different results, then the /// `true` results will still skip those particular items, while respecting the `false` /// result from elsewhere to skip any further items. /// /// This is similar to [`Iterator::skip_while`] without being constrained to the original /// iterator order. The remaining items will still maintain their relative order where that is /// visible in `collect`, `reduce`, and similar outputs. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..100) /// .into_par_iter() /// .skip_any_while(|x| *x < 50) /// .collect(); /// /// assert!(result.len() >= 50); /// assert!(result.windows(2).all(|w| w[0] < w[1])); /// ``` fn skip_any_while

(self, predicate: P) -> SkipAnyWhile where P: Fn(&Self::Item) -> bool + Sync + Send, { SkipAnyWhile::new(self, predicate) } /// Internal method used to define the behavior of this parallel /// iterator. You should not need to call this directly. /// /// This method causes the iterator `self` to start producing /// items and to feed them to the consumer `consumer` one by one. /// It may split the consumer before doing so to create the /// opportunity to produce in parallel. /// /// See the [README] for more details on the internals of parallel /// iterators. /// /// [README]: https://github.com/rayon-rs/rayon/blob/master/src/iter/plumbing/README.md fn drive_unindexed(self, consumer: C) -> C::Result where C: UnindexedConsumer; /// Internal method used to define the behavior of this parallel /// iterator. You should not need to call this directly. /// /// Returns the number of items produced by this iterator, if known /// statically. This can be used by consumers to trigger special fast /// paths. Therefore, if `Some(_)` is returned, this iterator must only /// use the (indexed) `Consumer` methods when driving a consumer, such /// as `split_at()`. Calling `UnindexedConsumer::split_off_left()` or /// other `UnindexedConsumer` methods -- or returning an inaccurate /// value -- may result in panics. /// /// This method is currently used to optimize `collect` for want /// of true Rust specialization; it may be removed when /// specialization is stable. fn opt_len(&self) -> Option { None } } impl IntoParallelIterator for T { type Iter = T; type Item = T::Item; fn into_par_iter(self) -> T { self } } /// An iterator that supports "random access" to its data, meaning /// that you can split it at arbitrary indices and draw data from /// those points. /// /// **Note:** Not implemented for `u64`, `i64`, `u128`, or `i128` ranges // Waiting for `ExactSizeIterator::is_empty` to be stabilized. See rust-lang/rust#35428 #[allow(clippy::len_without_is_empty)] pub trait IndexedParallelIterator: ParallelIterator { /// Collects the results of the iterator into the specified /// vector. The vector is always cleared before execution /// begins. If possible, reusing the vector across calls can lead /// to better performance since it reuses the same backing buffer. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// // any prior data will be cleared /// let mut vec = vec![-1, -2, -3]; /// /// (0..5).into_par_iter() /// .collect_into_vec(&mut vec); /// /// assert_eq!(vec, [0, 1, 2, 3, 4]); /// ``` fn collect_into_vec(self, target: &mut Vec) { collect::collect_into_vec(self, target); } /// Unzips the results of the iterator into the specified /// vectors. The vectors are always cleared before execution /// begins. If possible, reusing the vectors across calls can lead /// to better performance since they reuse the same backing buffer. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// // any prior data will be cleared /// let mut left = vec![42; 10]; /// let mut right = vec![-1; 10]; /// /// (10..15).into_par_iter() /// .enumerate() /// .unzip_into_vecs(&mut left, &mut right); /// /// assert_eq!(left, [0, 1, 2, 3, 4]); /// assert_eq!(right, [10, 11, 12, 13, 14]); /// ``` fn unzip_into_vecs(self, left: &mut Vec, right: &mut Vec) where Self: IndexedParallelIterator, A: Send, B: Send, { collect::unzip_into_vecs(self, left, right); } /// Iterates over tuples `(A, B)`, where the items `A` are from /// this iterator and `B` are from the iterator given as argument. /// Like the `zip` method on ordinary iterators, if the two /// iterators are of unequal length, you only get the items they /// have in common. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (1..4) /// .into_par_iter() /// .zip(vec!['a', 'b', 'c']) /// .collect(); /// /// assert_eq!(result, [(1, 'a'), (2, 'b'), (3, 'c')]); /// ``` fn zip(self, zip_op: Z) -> Zip where Z: IntoParallelIterator, Z::Iter: IndexedParallelIterator, { Zip::new(self, zip_op.into_par_iter()) } /// The same as `Zip`, but requires that both iterators have the same length. /// /// # Panics /// Will panic if `self` and `zip_op` are not the same length. /// /// ```should_panic /// use rayon::prelude::*; /// /// let one = [1u8]; /// let two = [2u8, 2]; /// let one_iter = one.par_iter(); /// let two_iter = two.par_iter(); /// /// // this will panic /// let zipped: Vec<(&u8, &u8)> = one_iter.zip_eq(two_iter).collect(); /// /// // we should never get here /// assert_eq!(1, zipped.len()); /// ``` #[track_caller] fn zip_eq(self, zip_op: Z) -> ZipEq where Z: IntoParallelIterator, Z::Iter: IndexedParallelIterator, { let zip_op_iter = zip_op.into_par_iter(); assert_eq!( self.len(), zip_op_iter.len(), "iterators must have the same length" ); ZipEq::new(self, zip_op_iter) } /// Interleaves elements of this iterator and the other given /// iterator. Alternately yields elements from this iterator and /// the given iterator, until both are exhausted. If one iterator /// is exhausted before the other, the last elements are provided /// from the other. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// let (x, y) = (vec![1, 2], vec![3, 4, 5, 6]); /// let r: Vec = x.into_par_iter().interleave(y).collect(); /// assert_eq!(r, vec![1, 3, 2, 4, 5, 6]); /// ``` fn interleave(self, other: I) -> Interleave where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, { Interleave::new(self, other.into_par_iter()) } /// Interleaves elements of this iterator and the other given /// iterator, until one is exhausted. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// let (x, y) = (vec![1, 2, 3, 4], vec![5, 6]); /// let r: Vec = x.into_par_iter().interleave_shortest(y).collect(); /// assert_eq!(r, vec![1, 5, 2, 6, 3]); /// ``` fn interleave_shortest(self, other: I) -> InterleaveShortest where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, { InterleaveShortest::new(self, other.into_par_iter()) } /// Splits an iterator up into fixed-size chunks. /// /// Returns an iterator that returns `Vec`s of the given number of elements. /// If the number of elements in the iterator is not divisible by `chunk_size`, /// the last chunk may be shorter than `chunk_size`. /// /// See also [`par_chunks()`] and [`par_chunks_mut()`] for similar behavior on /// slices, without having to allocate intermediate `Vec`s for the chunks. /// /// [`par_chunks()`]: ../slice/trait.ParallelSlice.html#method.par_chunks /// [`par_chunks_mut()`]: ../slice/trait.ParallelSliceMut.html#method.par_chunks_mut /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// let a = vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; /// let r: Vec> = a.into_par_iter().chunks(3).collect(); /// assert_eq!(r, vec![vec![1,2,3], vec![4,5,6], vec![7,8,9], vec![10]]); /// ``` #[track_caller] fn chunks(self, chunk_size: usize) -> Chunks { assert!(chunk_size != 0, "chunk_size must not be zero"); Chunks::new(self, chunk_size) } /// Splits an iterator into fixed-size chunks, performing a sequential [`fold()`] on /// each chunk. /// /// Returns an iterator that produces a folded result for each chunk of items /// produced by this iterator. /// /// This works essentially like: /// /// ```text /// iter.chunks(chunk_size) /// .map(|chunk| /// chunk.into_iter() /// .fold(identity, fold_op) /// ) /// ``` /// /// except there is no per-chunk allocation overhead. /// /// [`fold()`]: std::iter::Iterator#method.fold /// /// **Panics** if `chunk_size` is 0. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// let nums = vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; /// let chunk_sums = nums.into_par_iter().fold_chunks(2, || 0, |a, n| a + n).collect::>(); /// assert_eq!(chunk_sums, vec![3, 7, 11, 15, 19]); /// ``` #[track_caller] fn fold_chunks( self, chunk_size: usize, identity: ID, fold_op: F, ) -> FoldChunks where ID: Fn() -> T + Send + Sync, F: Fn(T, Self::Item) -> T + Send + Sync, T: Send, { assert!(chunk_size != 0, "chunk_size must not be zero"); FoldChunks::new(self, chunk_size, identity, fold_op) } /// Splits an iterator into fixed-size chunks, performing a sequential [`fold()`] on /// each chunk. /// /// Returns an iterator that produces a folded result for each chunk of items /// produced by this iterator. /// /// This works essentially like `fold_chunks(chunk_size, || init.clone(), fold_op)`, /// except it doesn't require the `init` type to be `Sync`, nor any other form of /// added synchronization. /// /// [`fold()`]: std::iter::Iterator#method.fold /// /// **Panics** if `chunk_size` is 0. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// let nums = vec![1, 2, 3, 4, 5, 6, 7, 8, 9, 10]; /// let chunk_sums = nums.into_par_iter().fold_chunks_with(2, 0, |a, n| a + n).collect::>(); /// assert_eq!(chunk_sums, vec![3, 7, 11, 15, 19]); /// ``` #[track_caller] fn fold_chunks_with( self, chunk_size: usize, init: T, fold_op: F, ) -> FoldChunksWith where T: Send + Clone, F: Fn(T, Self::Item) -> T + Send + Sync, { assert!(chunk_size != 0, "chunk_size must not be zero"); FoldChunksWith::new(self, chunk_size, init, fold_op) } /// Lexicographically compares the elements of this `ParallelIterator` with those of /// another. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use std::cmp::Ordering::*; /// /// let x = vec![1, 2, 3]; /// assert_eq!(x.par_iter().cmp(&vec![1, 3, 0]), Less); /// assert_eq!(x.par_iter().cmp(&vec![1, 2, 3]), Equal); /// assert_eq!(x.par_iter().cmp(&vec![1, 2]), Greater); /// ``` fn cmp(self, other: I) -> Ordering where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: Ord, { #[inline] fn ordering((x, y): (T, T)) -> Ordering { Ord::cmp(&x, &y) } #[inline] fn inequal(&ord: &Ordering) -> bool { ord != Ordering::Equal } let other = other.into_par_iter(); let ord_len = self.len().cmp(&other.len()); self.zip(other) .map(ordering) .find_first(inequal) .unwrap_or(ord_len) } /// Lexicographically compares the elements of this `ParallelIterator` with those of /// another. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use std::cmp::Ordering::*; /// use std::f64::NAN; /// /// let x = vec![1.0, 2.0, 3.0]; /// assert_eq!(x.par_iter().partial_cmp(&vec![1.0, 3.0, 0.0]), Some(Less)); /// assert_eq!(x.par_iter().partial_cmp(&vec![1.0, 2.0, 3.0]), Some(Equal)); /// assert_eq!(x.par_iter().partial_cmp(&vec![1.0, 2.0]), Some(Greater)); /// assert_eq!(x.par_iter().partial_cmp(&vec![1.0, NAN]), None); /// ``` fn partial_cmp(self, other: I) -> Option where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialOrd, { #[inline] fn ordering, U>((x, y): (T, U)) -> Option { PartialOrd::partial_cmp(&x, &y) } #[inline] fn inequal(&ord: &Option) -> bool { ord != Some(Ordering::Equal) } let other = other.into_par_iter(); let ord_len = self.len().cmp(&other.len()); self.zip(other) .map(ordering) .find_first(inequal) .unwrap_or(Some(ord_len)) } /// Determines if the elements of this `ParallelIterator` /// are equal to those of another fn eq(self, other: I) -> bool where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialEq, { #[inline] fn eq, U>((x, y): (T, U)) -> bool { PartialEq::eq(&x, &y) } let other = other.into_par_iter(); self.len() == other.len() && self.zip(other).all(eq) } /// Determines if the elements of this `ParallelIterator` /// are unequal to those of another fn ne(self, other: I) -> bool where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialEq, { !self.eq(other) } /// Determines if the elements of this `ParallelIterator` /// are lexicographically less than those of another. fn lt(self, other: I) -> bool where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialOrd, { self.partial_cmp(other) == Some(Ordering::Less) } /// Determines if the elements of this `ParallelIterator` /// are less or equal to those of another. fn le(self, other: I) -> bool where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialOrd, { let ord = self.partial_cmp(other); ord == Some(Ordering::Equal) || ord == Some(Ordering::Less) } /// Determines if the elements of this `ParallelIterator` /// are lexicographically greater than those of another. fn gt(self, other: I) -> bool where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialOrd, { self.partial_cmp(other) == Some(Ordering::Greater) } /// Determines if the elements of this `ParallelIterator` /// are less or equal to those of another. fn ge(self, other: I) -> bool where I: IntoParallelIterator, I::Iter: IndexedParallelIterator, Self::Item: PartialOrd, { let ord = self.partial_cmp(other); ord == Some(Ordering::Equal) || ord == Some(Ordering::Greater) } /// Yields an index along with each item. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let chars = vec!['a', 'b', 'c']; /// let result: Vec<_> = chars /// .into_par_iter() /// .enumerate() /// .collect(); /// /// assert_eq!(result, [(0, 'a'), (1, 'b'), (2, 'c')]); /// ``` fn enumerate(self) -> Enumerate { Enumerate::new(self) } /// Creates an iterator that steps by the given amount /// /// # Examples /// /// ``` ///use rayon::prelude::*; /// /// let range = (3..10); /// let result: Vec = range /// .into_par_iter() /// .step_by(3) /// .collect(); /// /// assert_eq!(result, [3, 6, 9]) /// ``` fn step_by(self, step: usize) -> StepBy { StepBy::new(self, step) } /// Creates an iterator that skips the first `n` elements. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..100) /// .into_par_iter() /// .skip(95) /// .collect(); /// /// assert_eq!(result, [95, 96, 97, 98, 99]); /// ``` fn skip(self, n: usize) -> Skip { Skip::new(self, n) } /// Creates an iterator that yields the first `n` elements. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..100) /// .into_par_iter() /// .take(5) /// .collect(); /// /// assert_eq!(result, [0, 1, 2, 3, 4]); /// ``` fn take(self, n: usize) -> Take { Take::new(self, n) } /// Searches for **some** item in the parallel iterator that /// matches the given predicate, and returns its index. Like /// `ParallelIterator::find_any`, the parallel search will not /// necessarily find the **first** match, and once a match is /// found we'll attempt to stop processing any more. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3, 3]; /// /// let i = a.par_iter().position_any(|&x| x == 3).expect("found"); /// assert!(i == 2 || i == 3); /// /// assert_eq!(a.par_iter().position_any(|&x| x == 100), None); /// ``` fn position_any

(self, predicate: P) -> Option where P: Fn(Self::Item) -> bool + Sync + Send, { #[inline] fn check(&(_, p): &(usize, bool)) -> bool { p } let (i, _) = self.map(predicate).enumerate().find_any(check)?; Some(i) } /// Searches for the sequentially **first** item in the parallel iterator /// that matches the given predicate, and returns its index. /// /// Like `ParallelIterator::find_first`, once a match is found, /// all attempts to the right of the match will be stopped, while /// attempts to the left must continue in case an earlier match /// is found. /// /// Note that not all parallel iterators have a useful order, much like /// sequential `HashMap` iteration, so "first" may be nebulous. If you /// just want the first match that discovered anywhere in the iterator, /// `position_any` is a better choice. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3, 3]; /// /// assert_eq!(a.par_iter().position_first(|&x| x == 3), Some(2)); /// /// assert_eq!(a.par_iter().position_first(|&x| x == 100), None); /// ``` fn position_first

(self, predicate: P) -> Option where P: Fn(Self::Item) -> bool + Sync + Send, { #[inline] fn check(&(_, p): &(usize, bool)) -> bool { p } let (i, _) = self.map(predicate).enumerate().find_first(check)?; Some(i) } /// Searches for the sequentially **last** item in the parallel iterator /// that matches the given predicate, and returns its index. /// /// Like `ParallelIterator::find_last`, once a match is found, /// all attempts to the left of the match will be stopped, while /// attempts to the right must continue in case a later match /// is found. /// /// Note that not all parallel iterators have a useful order, much like /// sequential `HashMap` iteration, so "last" may be nebulous. When the /// order doesn't actually matter to you, `position_any` is a better /// choice. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let a = [1, 2, 3, 3]; /// /// assert_eq!(a.par_iter().position_last(|&x| x == 3), Some(3)); /// /// assert_eq!(a.par_iter().position_last(|&x| x == 100), None); /// ``` fn position_last

(self, predicate: P) -> Option where P: Fn(Self::Item) -> bool + Sync + Send, { #[inline] fn check(&(_, p): &(usize, bool)) -> bool { p } let (i, _) = self.map(predicate).enumerate().find_last(check)?; Some(i) } #[doc(hidden)] #[deprecated( note = "parallel `position` does not search in order -- use `position_any`, \\ `position_first`, or `position_last`" )] fn position

(self, predicate: P) -> Option where P: Fn(Self::Item) -> bool + Sync + Send, { self.position_any(predicate) } /// Searches for items in the parallel iterator that match the given /// predicate, and returns their indices. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let primes = vec![2, 3, 5, 7, 11, 13, 17, 19, 23, 29]; /// /// // Find the positions of primes congruent to 1 modulo 6 /// let p1mod6: Vec<_> = primes.par_iter().positions(|&p| p % 6 == 1).collect(); /// assert_eq!(p1mod6, [3, 5, 7]); // primes 7, 13, and 19 /// /// // Find the positions of primes congruent to 5 modulo 6 /// let p5mod6: Vec<_> = primes.par_iter().positions(|&p| p % 6 == 5).collect(); /// assert_eq!(p5mod6, [2, 4, 6, 8, 9]); // primes 5, 11, 17, 23, and 29 /// ``` fn positions

(self, predicate: P) -> Positions where P: Fn(Self::Item) -> bool + Sync + Send, { Positions::new(self, predicate) } /// Produces a new iterator with the elements of this iterator in /// reverse order. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let result: Vec<_> = (0..5) /// .into_par_iter() /// .rev() /// .collect(); /// /// assert_eq!(result, [4, 3, 2, 1, 0]); /// ``` fn rev(self) -> Rev { Rev::new(self) } /// Sets the minimum length of iterators desired to process in each /// rayon job. Rayon will not split any smaller than this length, but /// of course an iterator could already be smaller to begin with. /// /// Producers like `zip` and `interleave` will use greater of the two /// minimums. /// Chained iterators and iterators inside `flat_map` may each use /// their own minimum length. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let min = (0..1_000_000) /// .into_par_iter() /// .with_min_len(1234) /// .fold(|| 0, |acc, _| acc + 1) // count how many are in this segment /// .min().unwrap(); /// /// assert!(min >= 1234); /// ``` fn with_min_len(self, min: usize) -> MinLen { MinLen::new(self, min) } /// Sets the maximum length of iterators desired to process in each /// rayon job. Rayon will try to split at least below this length, /// unless that would put it below the length from `with_min_len()`. /// For example, given min=10 and max=15, a length of 16 will not be /// split any further. /// /// Producers like `zip` and `interleave` will use lesser of the two /// maximums. /// Chained iterators and iterators inside `flat_map` may each use /// their own maximum length. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let max = (0..1_000_000) /// .into_par_iter() /// .with_max_len(1234) /// .fold(|| 0, |acc, _| acc + 1) // count how many are in this segment /// .max().unwrap(); /// /// assert!(max <= 1234); /// ``` fn with_max_len(self, max: usize) -> MaxLen { MaxLen::new(self, max) } /// Produces an exact count of how many items this iterator will /// produce, presuming no panic occurs. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let par_iter = (0..100).into_par_iter().zip(vec![0; 10]); /// assert_eq!(par_iter.len(), 10); /// /// let vec: Vec<_> = par_iter.collect(); /// assert_eq!(vec.len(), 10); /// ``` fn len(&self) -> usize; /// Internal method used to define the behavior of this parallel /// iterator. You should not need to call this directly. /// /// This method causes the iterator `self` to start producing /// items and to feed them to the consumer `consumer` one by one. /// It may split the consumer before doing so to create the /// opportunity to produce in parallel. If a split does happen, it /// will inform the consumer of the index where the split should /// occur (unlike `ParallelIterator::drive_unindexed()`). /// /// See the [README] for more details on the internals of parallel /// iterators. /// /// [README]: https://github.com/rayon-rs/rayon/blob/master/src/iter/plumbing/README.md fn drive>(self, consumer: C) -> C::Result; /// Internal method used to define the behavior of this parallel /// iterator. You should not need to call this directly. /// /// This method converts the iterator into a producer P and then /// invokes `callback.callback()` with P. Note that the type of /// this producer is not defined as part of the API, since /// `callback` must be defined generically for all producers. This /// allows the producer type to contain references; it also means /// that parallel iterators can adjust that type without causing a /// breaking change. /// /// See the [README] for more details on the internals of parallel /// iterators. /// /// [README]: https://github.com/rayon-rs/rayon/blob/master/src/iter/plumbing/README.md fn with_producer>(self, callback: CB) -> CB::Output; } /// `FromParallelIterator` implements the creation of a collection /// from a [`ParallelIterator`]. By implementing /// `FromParallelIterator` for a given type, you define how it will be /// created from an iterator. /// /// `FromParallelIterator` is used through [`ParallelIterator`]'s [`collect()`] method. /// /// [`ParallelIterator`]: trait.ParallelIterator.html /// [`collect()`]: trait.ParallelIterator.html#method.collect /// /// # Examples /// /// Implementing `FromParallelIterator` for your type: /// /// ``` /// use rayon::prelude::*; /// use std::mem; /// /// struct BlackHole { /// mass: usize, /// } /// /// impl FromParallelIterator for BlackHole { /// fn from_par_iter(par_iter: I) -> Self /// where I: IntoParallelIterator /// { /// let par_iter = par_iter.into_par_iter(); /// BlackHole { /// mass: par_iter.count() * mem::size_of::(), /// } /// } /// } /// /// let bh: BlackHole = (0i32..1000).into_par_iter().collect(); /// assert_eq!(bh.mass, 4000); /// ``` pub trait FromParallelIterator where T: Send, { /// Creates an instance of the collection from the parallel iterator `par_iter`. /// /// If your collection is not naturally parallel, the easiest (and /// fastest) way to do this is often to collect `par_iter` into a /// [`LinkedList`] or other intermediate data structure and then /// sequentially extend your collection. However, a more 'native' /// technique is to use the [`par_iter.fold`] or /// [`par_iter.fold_with`] methods to create the collection. /// Alternatively, if your collection is 'natively' parallel, you /// can use `par_iter.for_each` to process each element in turn. /// /// [`LinkedList`]: https://doc.rust-lang.org/std/collections/struct.LinkedList.html /// [`par_iter.fold`]: trait.ParallelIterator.html#method.fold /// [`par_iter.fold_with`]: trait.ParallelIterator.html#method.fold_with /// [`par_iter.for_each`]: trait.ParallelIterator.html#method.for_each fn from_par_iter(par_iter: I) -> Self where I: IntoParallelIterator; } /// `ParallelExtend` extends an existing collection with items from a [`ParallelIterator`]. /// /// [`ParallelIterator`]: trait.ParallelIterator.html /// /// # Examples /// /// Implementing `ParallelExtend` for your type: /// /// ``` /// use rayon::prelude::*; /// use std::mem; /// /// struct BlackHole { /// mass: usize, /// } /// /// impl ParallelExtend for BlackHole { /// fn par_extend(&mut self, par_iter: I) /// where I: IntoParallelIterator /// { /// let par_iter = par_iter.into_par_iter(); /// self.mass += par_iter.count() * mem::size_of::(); /// } /// } /// /// let mut bh = BlackHole { mass: 0 }; /// bh.par_extend(0i32..1000); /// assert_eq!(bh.mass, 4000); /// bh.par_extend(0i64..10); /// assert_eq!(bh.mass, 4080); /// ``` pub trait ParallelExtend where T: Send, { /// Extends an instance of the collection with the elements drawn /// from the parallel iterator `par_iter`. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let mut vec = vec![]; /// vec.par_extend(0..5); /// vec.par_extend((0..5).into_par_iter().map(|i| i * i)); /// assert_eq!(vec, [0, 1, 2, 3, 4, 0, 1, 4, 9, 16]); /// ``` fn par_extend(&mut self, par_iter: I) where I: IntoParallelIterator; } /// `ParallelDrainFull` creates a parallel iterator that moves all items /// from a collection while retaining the original capacity. /// /// Types which are indexable typically implement [`ParallelDrainRange`] /// instead, where you can drain fully with `par_drain(..)`. /// /// [`ParallelDrainRange`]: trait.ParallelDrainRange.html pub trait ParallelDrainFull { /// The draining parallel iterator type that will be created. type Iter: ParallelIterator; /// The type of item that the parallel iterator will produce. /// This is usually the same as `IntoParallelIterator::Item`. type Item: Send; /// Returns a draining parallel iterator over an entire collection. /// /// When the iterator is dropped, all items are removed, even if the /// iterator was not fully consumed. If the iterator is leaked, for example /// using `std::mem::forget`, it is unspecified how many items are removed. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// use std::collections::{BinaryHeap, HashSet}; /// /// let squares: HashSet = (0..10).map(|x| x * x).collect(); /// /// let mut heap: BinaryHeap<_> = squares.iter().copied().collect(); /// assert_eq!( /// // heaps are drained in arbitrary order /// heap.par_drain() /// .inspect(|x| assert!(squares.contains(x))) /// .count(), /// squares.len(), /// ); /// assert!(heap.is_empty()); /// assert!(heap.capacity() >= squares.len()); /// ``` fn par_drain(self) -> Self::Iter; } /// `ParallelDrainRange` creates a parallel iterator that moves a range of items /// from a collection while retaining the original capacity. /// /// Types which are not indexable may implement [`ParallelDrainFull`] instead. /// /// [`ParallelDrainFull`]: trait.ParallelDrainFull.html pub trait ParallelDrainRange { /// The draining parallel iterator type that will be created. type Iter: ParallelIterator; /// The type of item that the parallel iterator will produce. /// This is usually the same as `IntoParallelIterator::Item`. type Item: Send; /// Returns a draining parallel iterator over a range of the collection. /// /// When the iterator is dropped, all items in the range are removed, even /// if the iterator was not fully consumed. If the iterator is leaked, for /// example using `std::mem::forget`, it is unspecified how many items are /// removed. /// /// # Examples /// /// ``` /// use rayon::prelude::*; /// /// let squares: Vec = (0..10).map(|x| x * x).collect(); /// /// println!("RangeFull"); /// let mut vec = squares.clone(); /// assert!(vec.par_drain(..) /// .eq(squares.par_iter().copied())); /// assert!(vec.is_empty()); /// assert!(vec.capacity() >= squares.len()); /// /// println!("RangeFrom"); /// let mut vec = squares.clone(); /// assert!(vec.par_drain(5..) /// .eq(squares[5..].par_iter().copied())); /// assert_eq!(&vec[..], &squares[..5]); /// assert!(vec.capacity() >= squares.len()); /// /// println!("RangeTo"); /// let mut vec = squares.clone(); /// assert!(vec.par_drain(..5) /// .eq(squares[..5].par_iter().copied())); /// assert_eq!(&vec[..], &squares[5..]); /// assert!(vec.capacity() >= squares.len()); /// /// println!("RangeToInclusive"); /// let mut vec = squares.clone(); /// assert!(vec.par_drain(..=5) /// .eq(squares[..=5].par_iter().copied())); /// assert_eq!(&vec[..], &squares[6..]); /// assert!(vec.capacity() >= squares.len()); /// /// println!("Range"); /// let mut vec = squares.clone(); /// assert!(vec.par_drain(3..7) /// .eq(squares[3..7].par_iter().copied())); /// assert_eq!(&vec[..3], &squares[..3]); /// assert_eq!(&vec[3..], &squares[7..]); /// assert!(vec.capacity() >= squares.len()); /// /// println!("RangeInclusive"); /// let mut vec = squares.clone(); /// assert!(vec.par_drain(3..=7) /// .eq(squares[3..=7].par_iter().copied())); /// assert_eq!(&vec[..3], &squares[..3]); /// assert_eq!(&vec[3..], &squares[8..]); /// assert!(vec.capacity() >= squares.len()); /// ``` fn par_drain>(self, range: R) -> Self::Iter; } /// We hide the `Try` trait in a private module, as it's only meant to be a /// stable clone of the standard library's `Try` trait, as yet unstable. mod private { use std::convert::Infallible; use std::ops::ControlFlow::{self, Break, Continue}; use std::task::Poll; /// Clone of `std::ops::Try`. /// /// Implementing this trait is not permitted outside of `rayon`. pub trait Try { private_decl! {} type Output; type Residual; fn from_output(output: Self::Output) -> Self; fn from_residual(residual: Self::Residual) -> Self; fn branch(self) -> ControlFlow; } impl Try for ControlFlow { private_impl! {} type Output = C; type Residual = ControlFlow; fn from_output(output: Self::Output) -> Self { Continue(output) } fn from_residual(residual: Self::Residual) -> Self { match residual { Break(b) => Break(b), Continue(_) => unreachable!(), } } fn branch(self) -> ControlFlow { match self { Continue(c) => Continue(c), Break(b) => Break(Break(b)), } } } impl Try for Option { private_impl! {} type Output = T; type Residual = Option; fn from_output(output: Self::Output) -> Self { Some(output) } fn from_residual(residual: Self::Residual) -> Self { match residual { None => None, Some(_) => unreachable!(), } } fn branch(self) -> ControlFlow { match self { Some(c) => Continue(c), None => Break(None), } } } impl Try for Result { private_impl! {} type Output = T; type Residual = Result; fn from_output(output: Self::Output) -> Self { Ok(output) } fn from_residual(residual: Self::Residual) -> Self { match residual { Err(e) => Err(e), Ok(_) => unreachable!(), } } fn branch(self) -> ControlFlow { match self { Ok(c) => Continue(c), Err(e) => Break(Err(e)), } } } impl Try for Poll> { private_impl! {} type Output = Poll; type Residual = Result; fn from_output(output: Self::Output) -> Self { output.map(Ok) } fn from_residual(residual: Self::Residual) -> Self { match residual { Err(e) => Poll::Ready(Err(e)), Ok(_) => unreachable!(), } } fn branch(self) -> ControlFlow { match self { Poll::Pending => Continue(Poll::Pending), Poll::Ready(Ok(c)) => Continue(Poll::Ready(c)), Poll::Ready(Err(e)) => Break(Err(e)), } } } impl Try for Poll>> { private_impl! {} type Output = Poll>; type Residual = Result; fn from_output(output: Self::Output) -> Self { match output { Poll::Ready(o) => Poll::Ready(o.map(Ok)), Poll::Pending => Poll::Pending, } } fn from_residual(residual: Self::Residual) -> Self { match residual { Err(e) => Poll::Ready(Some(Err(e))), Ok(_) => unreachable!(), } } fn branch(self) -> ControlFlow { match self { Poll::Pending => Continue(Poll::Pending), Poll::Ready(None) => Continue(Poll::Ready(None)), Poll::Ready(Some(Ok(c))) => Continue(Poll::Ready(Some(c))), Poll::Ready(Some(Err(e))) => Break(Err(e)), } } } }