2014-01-11 14:13:06 +04:00
|
|
|
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
2013-04-09 10:54:32 -04:00
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2013-09-17 19:42:07 -07:00
|
|
|
/*!
|
2013-04-19 12:17:24 -04:00
|
|
|
|
2013-09-17 19:42:07 -07:00
|
|
|
Composable external iterators
|
2013-04-19 12:17:24 -04:00
|
|
|
|
2013-09-17 19:42:07 -07:00
|
|
|
# The `Iterator` trait
|
|
|
|
|
|
|
|
This module defines Rust's core iteration trait. The `Iterator` trait has one
|
2013-12-15 16:26:09 +11:00
|
|
|
unimplemented method, `next`. All other methods are derived through default
|
2013-09-17 19:42:07 -07:00
|
|
|
methods to perform operations such as `zip`, `chain`, `enumerate`, and `fold`.
|
|
|
|
|
|
|
|
The goal of this module is to unify iteration across all containers in Rust.
|
|
|
|
An iterator can be considered as a state machine which is used to track which
|
|
|
|
element will be yielded next.
|
|
|
|
|
|
|
|
There are various extensions also defined in this module to assist with various
|
|
|
|
types of iteration, such as the `DoubleEndedIterator` for iterating in reverse,
|
|
|
|
the `FromIterator` trait for creating a container from an iterator, and much
|
|
|
|
more.
|
|
|
|
|
|
|
|
## Rust's `for` loop
|
|
|
|
|
|
|
|
The special syntax used by rust's `for` loop is based around the `Iterator`
|
|
|
|
trait defined in this module. For loops can be viewed as a syntactical expansion
|
|
|
|
into a `loop`, for example, the `for` loop in this example is essentially
|
|
|
|
translated to the `loop` below.
|
|
|
|
|
2013-09-23 17:20:36 -07:00
|
|
|
```rust
|
2013-09-17 19:42:07 -07:00
|
|
|
let values = ~[1, 2, 3];
|
|
|
|
|
|
|
|
// "Syntactical sugar" taking advantage of an iterator
|
|
|
|
for &x in values.iter() {
|
|
|
|
println!("{}", x);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rough translation of the iteration without a `for` iterator.
|
|
|
|
let mut it = values.iter();
|
|
|
|
loop {
|
|
|
|
match it.next() {
|
|
|
|
Some(&x) => {
|
|
|
|
println!("{}", x);
|
|
|
|
}
|
|
|
|
None => { break }
|
|
|
|
}
|
|
|
|
}
|
2014-05-02 17:56:35 -07:00
|
|
|
```
|
2013-09-17 19:42:07 -07:00
|
|
|
|
|
|
|
This `for` loop syntax can be applied to any iterator over any type.
|
|
|
|
|
|
|
|
## Iteration protocol and more
|
|
|
|
|
|
|
|
More detailed information about iterators can be found in the [container
|
2014-02-11 14:38:36 +00:00
|
|
|
guide](http://static.rust-lang.org/doc/master/guide-container.html) with
|
2013-09-17 19:42:07 -07:00
|
|
|
the rest of the rust manuals.
|
2013-04-19 12:17:24 -04:00
|
|
|
|
|
|
|
*/
|
2013-04-09 10:54:32 -04:00
|
|
|
|
2013-05-24 19:35:29 -07:00
|
|
|
use cmp;
|
2014-02-17 07:20:01 +11:00
|
|
|
use num::{Zero, One, CheckedAdd, CheckedSub, Saturating, ToPrimitive, Int};
|
2013-06-06 22:34:50 +02:00
|
|
|
use option::{Option, Some, None};
|
2013-08-06 22:34:22 -07:00
|
|
|
use ops::{Add, Mul, Sub};
|
2014-03-08 01:10:32 -05:00
|
|
|
use cmp::{Eq, Ord, TotalOrd};
|
2013-06-06 22:34:50 +02:00
|
|
|
use clone::Clone;
|
2013-07-02 20:59:26 -07:00
|
|
|
use uint;
|
2014-02-01 04:35:36 +08:00
|
|
|
use mem;
|
2013-04-09 10:54:32 -04:00
|
|
|
|
2013-06-21 07:57:22 -04:00
|
|
|
/// Conversion from an `Iterator`
|
std: Move the iterator param on FromIterator and Extendable to the method.
If they are on the trait then it is extremely annoying to use them as
generic parameters to a function, e.g. with the iterator param on the trait
itself, if one was to pass an Extendable<int> to a function that filled it
either from a Range or a Map<VecIterator>, one needs to write something
like:
fn foo<E: Extendable<int, Range<int>> +
Extendable<int, Map<&'self int, int, VecIterator<int>>>
(e: &mut E, ...) { ... }
since using a generic, i.e. `foo<E: Extendable<int, I>, I: Iterator<int>>`
means that `foo` takes 2 type parameters, and the caller has to specify them
(which doesn't work anyway, as they'll mismatch with the iterators used in
`foo` itself).
This patch changes it to:
fn foo<E: Extendable<int>>(e: &mut E, ...) { ... }
2013-08-13 23:08:14 +10:00
|
|
|
pub trait FromIterator<A> {
|
2013-06-21 07:57:22 -04:00
|
|
|
/// Build a container with elements from an external iterator.
|
2014-03-30 21:45:55 -07:00
|
|
|
fn from_iter<T: Iterator<A>>(iterator: T) -> Self;
|
2013-06-21 07:57:22 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 17:41:30 -04:00
|
|
|
/// A type growable from an `Iterator` implementation
|
std: Move the iterator param on FromIterator and Extendable to the method.
If they are on the trait then it is extremely annoying to use them as
generic parameters to a function, e.g. with the iterator param on the trait
itself, if one was to pass an Extendable<int> to a function that filled it
either from a Range or a Map<VecIterator>, one needs to write something
like:
fn foo<E: Extendable<int, Range<int>> +
Extendable<int, Map<&'self int, int, VecIterator<int>>>
(e: &mut E, ...) { ... }
since using a generic, i.e. `foo<E: Extendable<int, I>, I: Iterator<int>>`
means that `foo` takes 2 type parameters, and the caller has to specify them
(which doesn't work anyway, as they'll mismatch with the iterators used in
`foo` itself).
This patch changes it to:
fn foo<E: Extendable<int>>(e: &mut E, ...) { ... }
2013-08-13 23:08:14 +10:00
|
|
|
pub trait Extendable<A>: FromIterator<A> {
|
2013-07-27 17:41:30 -04:00
|
|
|
/// Extend a container with the elements yielded by an iterator
|
2014-03-20 14:12:56 +01:00
|
|
|
fn extend<T: Iterator<A>>(&mut self, iterator: T);
|
2013-07-27 17:41:30 -04:00
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An interface for dealing with "external iterators". These types of iterators
|
|
|
|
/// can be resumed at any time as all state is stored internally as opposed to
|
|
|
|
/// being located on the call stack.
|
2013-08-03 13:51:49 -07:00
|
|
|
///
|
|
|
|
/// The Iterator protocol states that an iterator yields a (potentially-empty,
|
|
|
|
/// potentially-infinite) sequence of values, and returns `None` to signal that
|
|
|
|
/// it's finished. The Iterator protocol does not define behavior after `None`
|
|
|
|
/// is returned. A concrete Iterator implementation may choose to behave however
|
|
|
|
/// it wishes, either by returning `None` infinitely, or by doing something
|
|
|
|
/// else.
|
2013-04-15 10:30:16 -04:00
|
|
|
pub trait Iterator<A> {
|
2013-04-09 10:54:32 -04:00
|
|
|
/// Advance the iterator and return the next value. Return `None` when the end is reached.
|
2013-04-15 10:30:16 -04:00
|
|
|
fn next(&mut self) -> Option<A>;
|
2013-06-21 06:12:01 -04:00
|
|
|
|
|
|
|
/// Return a lower bound and upper bound on the remaining length of the iterator.
|
|
|
|
///
|
|
|
|
/// The common use case for the estimate is pre-allocating space to store the results.
|
2013-08-01 18:35:46 -04:00
|
|
|
#[inline]
|
2013-07-02 18:40:46 -07:00
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) { (0, None) }
|
2013-07-11 00:23:59 -04:00
|
|
|
|
2013-06-11 16:34:03 -07:00
|
|
|
/// Chain this iterator with another, returning a new iterator which will
|
2013-05-28 16:35:52 -05:00
|
|
|
/// finish iterating over the current iterator, and then it will iterate
|
|
|
|
/// over the other specified iterator.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [0];
|
|
|
|
/// let b = [1];
|
2013-08-09 20:22:59 -07:00
|
|
|
/// let mut it = a.iter().chain(b.iter());
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), &0);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &1);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-08-09 20:22:59 -07:00
|
|
|
fn chain<U: Iterator<A>>(self, other: U) -> Chain<Self, U> {
|
2013-08-09 07:19:23 -07:00
|
|
|
Chain{a: self, b: other, flag: false}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates an iterator which iterates over both this and the specified
|
|
|
|
/// iterators simultaneously, yielding the two elements as pairs. When
|
|
|
|
/// either iterator returns None, all further invocations of next() will
|
|
|
|
/// return None.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [0];
|
|
|
|
/// let b = [1];
|
|
|
|
/// let mut it = a.iter().zip(b.iter());
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), (&0, &1));
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn zip<B, U: Iterator<B>>(self, other: U) -> Zip<Self, U> {
|
|
|
|
Zip{a: self, b: other}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates a new iterator which will apply the specified function to each
|
2013-06-29 15:05:50 +10:00
|
|
|
/// element returned by the first, yielding the mapped element instead.
|
2013-05-28 16:35:52 -05:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2];
|
2013-08-09 20:09:47 -07:00
|
|
|
/// let mut it = a.iter().map(|&x| 2 * x);
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), 2);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 4);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn map<'r, B>(self, f: |A|: 'r -> B) -> Map<'r, A, B, Self> {
|
2013-08-09 07:19:23 -07:00
|
|
|
Map{iter: self, f: f}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates an iterator which applies the predicate to each element returned
|
|
|
|
/// by this iterator. Only elements which have the predicate evaluate to
|
|
|
|
/// `true` will be yielded.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2];
|
|
|
|
/// let mut it = a.iter().filter(|&x| *x > 1);
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), &2);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn filter<'r>(self, predicate: |&A|: 'r -> bool) -> Filter<'r, A, Self> {
|
2013-08-09 07:19:23 -07:00
|
|
|
Filter{iter: self, predicate: predicate}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
2013-06-11 16:34:03 -07:00
|
|
|
/// Creates an iterator which both filters and maps elements.
|
2013-05-28 16:35:52 -05:00
|
|
|
/// If the specified function returns None, the element is skipped.
|
|
|
|
/// Otherwise the option is unwrapped and the new value is yielded.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2];
|
|
|
|
/// let mut it = a.iter().filter_map(|&x| if x > 1 {Some(2 * x)} else {None});
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), 4);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn filter_map<'r, B>(self, f: |A|: 'r -> Option<B>) -> FilterMap<'r, A, B, Self> {
|
2013-08-09 07:19:23 -07:00
|
|
|
FilterMap { iter: self, f: f }
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates an iterator which yields a pair of the value returned by this
|
|
|
|
/// iterator plus the current index of iteration.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [100, 200];
|
|
|
|
/// let mut it = a.iter().enumerate();
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), (0, &100));
|
|
|
|
/// assert_eq!(it.next().unwrap(), (1, &200));
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn enumerate(self) -> Enumerate<Self> {
|
|
|
|
Enumerate{iter: self, count: 0}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
2013-08-08 20:46:36 +01:00
|
|
|
|
|
|
|
/// Creates an iterator that has a `.peek()` method
|
2014-01-30 19:29:35 +01:00
|
|
|
/// that returns an optional reference to the next element.
|
2013-08-08 20:46:36 +01:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// let xs = [100, 200, 300];
|
|
|
|
/// let mut it = xs.iter().map(|x| *x).peekable();
|
2013-08-08 20:46:36 +01:00
|
|
|
/// assert_eq!(it.peek().unwrap(), &100);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 100);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 200);
|
|
|
|
/// assert_eq!(it.peek().unwrap(), &300);
|
|
|
|
/// assert_eq!(it.peek().unwrap(), &300);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 300);
|
|
|
|
/// assert!(it.peek().is_none());
|
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-13 21:41:50 -04:00
|
|
|
#[inline]
|
2013-08-08 20:46:36 +01:00
|
|
|
fn peekable(self) -> Peekable<A, Self> {
|
|
|
|
Peekable{iter: self, peeked: None}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Creates an iterator which invokes the predicate on elements until it
|
2013-06-11 16:34:03 -07:00
|
|
|
/// returns false. Once the predicate returns false, all further elements are
|
2013-05-28 16:35:52 -05:00
|
|
|
/// yielded.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 2, 1];
|
|
|
|
/// let mut it = a.iter().skip_while(|&a| *a < 3);
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), &3);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &2);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &1);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn skip_while<'r>(self, predicate: |&A|: 'r -> bool) -> SkipWhile<'r, A, Self> {
|
2013-08-09 07:19:23 -07:00
|
|
|
SkipWhile{iter: self, flag: false, predicate: predicate}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates an iterator which yields elements so long as the predicate
|
|
|
|
/// returns true. After the predicate returns false for the first time, no
|
|
|
|
/// further elements will be yielded.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 2, 1];
|
|
|
|
/// let mut it = a.iter().take_while(|&a| *a < 3);
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), &1);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &2);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn take_while<'r>(self, predicate: |&A|: 'r -> bool) -> TakeWhile<'r, A, Self> {
|
2013-08-09 07:19:23 -07:00
|
|
|
TakeWhile{iter: self, flag: false, predicate: predicate}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates an iterator which skips the first `n` elements of this iterator,
|
|
|
|
/// and then it yields all further items.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
|
|
|
/// let mut it = a.iter().skip(3);
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), &4);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &5);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn skip(self, n: uint) -> Skip<Self> {
|
|
|
|
Skip{iter: self, n: n}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Creates an iterator which yields the first `n` elements of this
|
|
|
|
/// iterator, and then it will always return None.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-08-09 20:16:07 -07:00
|
|
|
/// let mut it = a.iter().take(3);
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), &1);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &2);
|
|
|
|
/// assert_eq!(it.next().unwrap(), &3);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-08-09 20:16:07 -07:00
|
|
|
fn take(self, n: uint) -> Take<Self> {
|
2013-08-09 07:19:23 -07:00
|
|
|
Take{iter: self, n: n}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
2014-04-27 16:50:55 +02:00
|
|
|
/// Creates a new iterator which behaves in a similar fashion to fold.
|
2013-05-28 16:35:52 -05:00
|
|
|
/// There is a state which is passed between each iteration and can be
|
|
|
|
/// mutated as necessary. The yielded values from the closure are yielded
|
2013-07-27 23:41:20 +02:00
|
|
|
/// from the Scan instance when not None.
|
2013-05-28 16:35:52 -05:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
|
|
|
/// let mut it = a.iter().scan(1, |fac, &x| {
|
|
|
|
/// *fac = *fac * x;
|
|
|
|
/// Some(*fac)
|
|
|
|
/// });
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert_eq!(it.next().unwrap(), 1);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 2);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 6);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 24);
|
|
|
|
/// assert_eq!(it.next().unwrap(), 120);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.next().is_none());
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn scan<'r, St, B>(self, initial_state: St, f: |&mut St, A|: 'r -> Option<B>)
|
2013-08-09 07:19:23 -07:00
|
|
|
-> Scan<'r, A, B, Self, St> {
|
|
|
|
Scan{iter: self, f: f, state: initial_state}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
2013-06-24 02:34:38 +02:00
|
|
|
/// Creates an iterator that maps each element to an iterator,
|
|
|
|
/// and yields the elements of the produced iterators
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// use std::iter::count;
|
|
|
|
///
|
2013-06-24 02:34:38 +02:00
|
|
|
/// let xs = [2u, 3];
|
|
|
|
/// let ys = [0u, 1, 0, 1, 2];
|
2013-08-09 20:20:05 -07:00
|
|
|
/// let mut it = xs.iter().flat_map(|&x| count(0u, 1).take(x));
|
2013-06-24 02:34:38 +02:00
|
|
|
/// // Check that `it` has the same elements as `ys`
|
|
|
|
/// let mut i = 0;
|
2013-12-22 13:31:23 -08:00
|
|
|
/// for x in it {
|
2013-06-24 02:34:38 +02:00
|
|
|
/// assert_eq!(x, ys[i]);
|
|
|
|
/// i += 1;
|
|
|
|
/// }
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn flat_map<'r, B, U: Iterator<B>>(self, f: |A|: 'r -> U)
|
2013-08-09 07:19:23 -07:00
|
|
|
-> FlatMap<'r, A, Self, U> {
|
|
|
|
FlatMap{iter: self, f: f, frontiter: None, backiter: None }
|
|
|
|
}
|
2013-06-24 02:34:38 +02:00
|
|
|
|
2013-08-03 13:51:49 -07:00
|
|
|
/// Creates an iterator that yields `None` forever after the underlying
|
|
|
|
/// iterator yields `None`. Random-access iterator behavior is not
|
|
|
|
/// affected, only single and double-ended iterator behavior.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-08-03 13:51:49 -07:00
|
|
|
/// fn process<U: Iterator<int>>(it: U) -> int {
|
|
|
|
/// let mut it = it.fuse();
|
|
|
|
/// let mut sum = 0;
|
|
|
|
/// for x in it {
|
|
|
|
/// if x > 5 {
|
2013-12-22 13:31:23 -08:00
|
|
|
/// continue;
|
2013-08-03 13:51:49 -07:00
|
|
|
/// }
|
|
|
|
/// sum += x;
|
|
|
|
/// }
|
|
|
|
/// // did we exhaust the iterator?
|
|
|
|
/// if it.next().is_none() {
|
|
|
|
/// sum += 1000;
|
|
|
|
/// }
|
|
|
|
/// sum
|
|
|
|
/// }
|
|
|
|
/// let x = ~[1,2,3,7,8,9];
|
|
|
|
/// assert_eq!(process(x.move_iter()), 1006);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-03 13:51:49 -07:00
|
|
|
#[inline]
|
|
|
|
fn fuse(self) -> Fuse<Self> {
|
|
|
|
Fuse{iter: self, done: false}
|
|
|
|
}
|
|
|
|
|
2013-07-05 13:00:11 -04:00
|
|
|
/// Creates an iterator that calls a function with a reference to each
|
|
|
|
/// element before yielding it. This is often useful for debugging an
|
|
|
|
/// iterator pipeline.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// use std::iter::AdditiveIterator;
|
|
|
|
///
|
|
|
|
/// let xs = [1u, 4, 2, 3, 8, 9, 6];
|
|
|
|
/// let sum = xs.iter()
|
|
|
|
/// .map(|&x| x)
|
log: Introduce liblog, the old std::logging
This commit moves all logging out of the standard library into an external
crate. This crate is the new crate which is responsible for all logging macros
and logging implementation. A few reasons for this change are:
* The crate map has always been a bit of a code smell among rust programs. It
has difficulty being loaded on almost all platforms, and it's used almost
exclusively for logging and only logging. Removing the crate map is one of the
end goals of this movement.
* The compiler has a fair bit of special support for logging. It has the
__log_level() expression as well as generating a global word per module
specifying the log level. This is unfairly favoring the built-in logging
system, and is much better done purely in libraries instead of the compiler
itself.
* Initialization of logging is much easier to do if there is no reliance on a
magical crate map being available to set module log levels.
* If the logging library can be written outside of the standard library, there's
no reason that it shouldn't be. It's likely that we're not going to build the
highest quality logging library of all time, so third-party libraries should
be able to provide just as high-quality logging systems as the default one
provided in the rust distribution.
With a migration such as this, the change does not come for free. There are some
subtle changes in the behavior of liblog vs the previous logging macros:
* The core change of this migration is that there is no longer a physical
log-level per module. This concept is still emulated (it is quite useful), but
there is now only a global log level, not a local one. This global log level
is a reflection of the maximum of all log levels specified. The previously
generated logging code looked like:
if specified_level <= __module_log_level() {
println!(...)
}
The newly generated code looks like:
if specified_level <= ::log::LOG_LEVEL {
if ::log::module_enabled(module_path!()) {
println!(...)
}
}
Notably, the first layer of checking is still intended to be "super fast" in
that it's just a load of a global word and a compare. The second layer of
checking is executed to determine if the current module does indeed have
logging turned on.
This means that if any module has a debug log level turned on, all modules
with debug log levels get a little bit slower (they all do more expensive
dynamic checks to determine if they're turned on or not).
Semantically, this migration brings no change in this respect, but
runtime-wise, this will have a perf impact on some code.
* A `RUST_LOG=::help` directive will no longer print out a list of all modules
that can be logged. This is because the crate map will no longer specify the
log levels of all modules, so the list of modules is not known. Additionally,
warnings can no longer be provided if a malformed logging directive was
supplied.
The new "hello world" for logging looks like:
#[phase(syntax, link)]
extern crate log;
fn main() {
debug!("Hello, world!");
}
2014-03-08 22:11:44 -08:00
|
|
|
/// .inspect(|&x| println!("filtering {}", x))
|
2013-12-22 13:31:23 -08:00
|
|
|
/// .filter(|&x| x % 2 == 0)
|
log: Introduce liblog, the old std::logging
This commit moves all logging out of the standard library into an external
crate. This crate is the new crate which is responsible for all logging macros
and logging implementation. A few reasons for this change are:
* The crate map has always been a bit of a code smell among rust programs. It
has difficulty being loaded on almost all platforms, and it's used almost
exclusively for logging and only logging. Removing the crate map is one of the
end goals of this movement.
* The compiler has a fair bit of special support for logging. It has the
__log_level() expression as well as generating a global word per module
specifying the log level. This is unfairly favoring the built-in logging
system, and is much better done purely in libraries instead of the compiler
itself.
* Initialization of logging is much easier to do if there is no reliance on a
magical crate map being available to set module log levels.
* If the logging library can be written outside of the standard library, there's
no reason that it shouldn't be. It's likely that we're not going to build the
highest quality logging library of all time, so third-party libraries should
be able to provide just as high-quality logging systems as the default one
provided in the rust distribution.
With a migration such as this, the change does not come for free. There are some
subtle changes in the behavior of liblog vs the previous logging macros:
* The core change of this migration is that there is no longer a physical
log-level per module. This concept is still emulated (it is quite useful), but
there is now only a global log level, not a local one. This global log level
is a reflection of the maximum of all log levels specified. The previously
generated logging code looked like:
if specified_level <= __module_log_level() {
println!(...)
}
The newly generated code looks like:
if specified_level <= ::log::LOG_LEVEL {
if ::log::module_enabled(module_path!()) {
println!(...)
}
}
Notably, the first layer of checking is still intended to be "super fast" in
that it's just a load of a global word and a compare. The second layer of
checking is executed to determine if the current module does indeed have
logging turned on.
This means that if any module has a debug log level turned on, all modules
with debug log levels get a little bit slower (they all do more expensive
dynamic checks to determine if they're turned on or not).
Semantically, this migration brings no change in this respect, but
runtime-wise, this will have a perf impact on some code.
* A `RUST_LOG=::help` directive will no longer print out a list of all modules
that can be logged. This is because the crate map will no longer specify the
log levels of all modules, so the list of modules is not known. Additionally,
warnings can no longer be provided if a malformed logging directive was
supplied.
The new "hello world" for logging looks like:
#[phase(syntax, link)]
extern crate log;
fn main() {
debug!("Hello, world!");
}
2014-03-08 22:11:44 -08:00
|
|
|
/// .inspect(|&x| println!("{} made it through", x))
|
2013-12-22 13:31:23 -08:00
|
|
|
/// .sum();
|
2014-01-09 21:06:55 +11:00
|
|
|
/// println!("{}", sum);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
fn inspect<'r>(self, f: |&A|: 'r) -> Inspect<'r, A, Self> {
|
2013-08-11 01:55:34 +02:00
|
|
|
Inspect{iter: self, f: f}
|
2013-08-09 07:19:23 -07:00
|
|
|
}
|
2013-07-05 13:00:11 -04:00
|
|
|
|
2013-09-30 20:59:46 +02:00
|
|
|
/// Creates a wrapper around a mutable reference to the iterator.
|
|
|
|
///
|
|
|
|
/// This is useful to allow applying iterator adaptors while still
|
|
|
|
/// retaining ownership of the original iterator value.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// let mut xs = range(0, 10);
|
|
|
|
/// // sum the first five values
|
|
|
|
/// let partial_sum = xs.by_ref().take(5).fold(0, |a, b| a + b);
|
|
|
|
/// assert!(partial_sum == 10);
|
|
|
|
/// // xs.next() is now `5`
|
|
|
|
/// assert!(xs.next() == Some(5));
|
|
|
|
/// ```
|
|
|
|
fn by_ref<'r>(&'r mut self) -> ByRef<'r, Self> {
|
|
|
|
ByRef{iter: self}
|
|
|
|
}
|
|
|
|
|
2013-11-17 15:37:00 -03:00
|
|
|
/// Apply a function to each element, or stop iterating if the
|
|
|
|
/// function returns `false`.
|
2013-05-28 16:35:52 -05:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-11-17 15:37:00 -03:00
|
|
|
/// range(0, 5).advance(|x| {print!("{} ", x); true});
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn advance(&mut self, f: |A| -> bool) -> bool {
|
2013-08-09 07:19:23 -07:00
|
|
|
loop {
|
|
|
|
match self.next() {
|
|
|
|
Some(x) => {
|
|
|
|
if !f(x) { return false; }
|
|
|
|
}
|
|
|
|
None => { return true; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
2013-06-03 23:48:52 +02:00
|
|
|
/// Loops through the entire iterator, collecting all of the elements into
|
2013-06-24 01:21:32 -04:00
|
|
|
/// a container implementing `FromIterator`.
|
2013-06-03 23:48:52 +02:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-06-03 23:48:52 +02:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-08-09 20:09:47 -07:00
|
|
|
/// let b: ~[int] = a.iter().map(|&x| x).collect();
|
2013-06-03 23:48:52 +02:00
|
|
|
/// assert!(a == b);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
std: Move the iterator param on FromIterator and Extendable to the method.
If they are on the trait then it is extremely annoying to use them as
generic parameters to a function, e.g. with the iterator param on the trait
itself, if one was to pass an Extendable<int> to a function that filled it
either from a Range or a Map<VecIterator>, one needs to write something
like:
fn foo<E: Extendable<int, Range<int>> +
Extendable<int, Map<&'self int, int, VecIterator<int>>>
(e: &mut E, ...) { ... }
since using a generic, i.e. `foo<E: Extendable<int, I>, I: Iterator<int>>`
means that `foo` takes 2 type parameters, and the caller has to specify them
(which doesn't work anyway, as they'll mismatch with the iterators used in
`foo` itself).
This patch changes it to:
fn foo<E: Extendable<int>>(e: &mut E, ...) { ... }
2013-08-13 23:08:14 +10:00
|
|
|
fn collect<B: FromIterator<A>>(&mut self) -> B {
|
2014-03-30 21:45:55 -07:00
|
|
|
FromIterator::from_iter(self.by_ref())
|
2013-08-09 07:19:23 -07:00
|
|
|
}
|
2013-06-03 23:48:52 +02:00
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Loops through `n` iterations, returning the `n`th element of the
|
|
|
|
/// iterator.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
|
|
|
/// let mut it = a.iter();
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert!(it.nth(2).unwrap() == &3);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.nth(2) == None);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn nth(&mut self, mut n: uint) -> Option<A> {
|
|
|
|
loop {
|
|
|
|
match self.next() {
|
|
|
|
Some(x) => if n == 0 { return Some(x) },
|
|
|
|
None => return None
|
|
|
|
}
|
|
|
|
n -= 1;
|
|
|
|
}
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Loops through the entire iterator, returning the last element of the
|
|
|
|
/// iterator.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert!(a.iter().last().unwrap() == &5);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-08-09 20:25:45 -07:00
|
|
|
fn last(&mut self) -> Option<A> {
|
2013-08-09 07:19:23 -07:00
|
|
|
let mut last = None;
|
|
|
|
for x in *self { last = Some(x); }
|
|
|
|
last
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Performs a fold operation over the entire iterator, returning the
|
|
|
|
/// eventual state at the end of the iteration.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
|
|
|
/// assert!(a.iter().fold(0, |a, &b| a + b) == 15);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn fold<B>(&mut self, init: B, f: |B, A| -> B) -> B {
|
2013-08-09 07:19:23 -07:00
|
|
|
let mut accum = init;
|
|
|
|
loop {
|
|
|
|
match self.next() {
|
|
|
|
Some(x) => { accum = f(accum, x); }
|
|
|
|
None => { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
accum
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Counts the number of elements in this iterator.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
|
|
|
/// let mut it = a.iter();
|
2013-08-09 20:30:03 -07:00
|
|
|
/// assert!(it.len() == 5);
|
|
|
|
/// assert!(it.len() == 0);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-08-09 20:30:03 -07:00
|
|
|
fn len(&mut self) -> uint {
|
2013-08-09 07:19:23 -07:00
|
|
|
self.fold(0, |cnt, _x| cnt + 1)
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Tests whether the predicate holds true for all elements in the iterator.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-12-22 13:31:23 -08:00
|
|
|
/// assert!(a.iter().all(|x| *x > 0));
|
|
|
|
/// assert!(!a.iter().all(|x| *x > 2));
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn all(&mut self, f: |A| -> bool) -> bool {
|
2013-08-09 07:19:23 -07:00
|
|
|
for x in *self { if !f(x) { return false; } }
|
|
|
|
true
|
|
|
|
}
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Tests whether any element of an iterator satisfies the specified
|
|
|
|
/// predicate.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
|
|
|
/// let mut it = a.iter();
|
2013-12-22 13:31:23 -08:00
|
|
|
/// assert!(it.any(|x| *x == 3));
|
|
|
|
/// assert!(!it.any(|x| *x == 3));
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn any(&mut self, f: |A| -> bool) -> bool {
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in *self { if f(x) { return true; } }
|
2013-06-15 17:42:31 -04:00
|
|
|
false
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the first element satisfying the specified predicate
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn find(&mut self, predicate: |&A| -> bool) -> Option<A> {
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in *self {
|
2013-06-15 17:42:31 -04:00
|
|
|
if predicate(&x) { return Some(x) }
|
|
|
|
}
|
|
|
|
None
|
2013-05-18 00:24:43 +09:00
|
|
|
}
|
2013-06-15 17:56:26 -04:00
|
|
|
|
|
|
|
/// Return the index of the first element satisfying the specified predicate
|
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn position(&mut self, predicate: |A| -> bool) -> Option<uint> {
|
2013-06-15 17:56:26 -04:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in *self {
|
2013-06-15 17:56:26 -04:00
|
|
|
if predicate(x) {
|
|
|
|
return Some(i);
|
|
|
|
}
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
2013-06-17 19:43:22 -04:00
|
|
|
|
2013-08-09 07:19:23 -07:00
|
|
|
/// Count the number of elements satisfying the specified predicate
|
2013-06-17 19:43:22 -04:00
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn count(&mut self, predicate: |A| -> bool) -> uint {
|
2013-06-17 19:43:22 -04:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in *self {
|
2013-06-17 19:43:22 -04:00
|
|
|
if predicate(x) { i += 1 }
|
|
|
|
}
|
|
|
|
i
|
|
|
|
}
|
2013-06-27 06:51:04 +09:00
|
|
|
|
2013-08-18 08:28:04 +10:00
|
|
|
/// Return the element that gives the maximum value from the
|
|
|
|
/// specified function.
|
2013-08-09 07:19:23 -07:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// let xs = [-3i, 0, 1, 5, -10];
|
2013-08-09 07:19:23 -07:00
|
|
|
/// assert_eq!(*xs.iter().max_by(|x| x.abs()).unwrap(), -10);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-06-27 06:51:04 +09:00
|
|
|
#[inline]
|
2014-03-08 01:10:32 -05:00
|
|
|
fn max_by<B: TotalOrd>(&mut self, f: |&A| -> B) -> Option<A> {
|
2013-06-27 06:51:04 +09:00
|
|
|
self.fold(None, |max: Option<(A, B)>, x| {
|
|
|
|
let x_val = f(&x);
|
|
|
|
match max {
|
|
|
|
None => Some((x, x_val)),
|
|
|
|
Some((y, y_val)) => if x_val > y_val {
|
|
|
|
Some((x, x_val))
|
|
|
|
} else {
|
|
|
|
Some((y, y_val))
|
|
|
|
}
|
|
|
|
}
|
2013-09-20 02:08:47 -04:00
|
|
|
}).map(|(x, _)| x)
|
2013-06-27 06:51:04 +09:00
|
|
|
}
|
|
|
|
|
2013-08-18 08:28:04 +10:00
|
|
|
/// Return the element that gives the minimum value from the
|
|
|
|
/// specified function.
|
2013-08-09 07:19:23 -07:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// let xs = [-3i, 0, 1, 5, -10];
|
2013-08-09 07:19:23 -07:00
|
|
|
/// assert_eq!(*xs.iter().min_by(|x| x.abs()).unwrap(), 0);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-06-27 06:51:04 +09:00
|
|
|
#[inline]
|
2014-03-08 01:10:32 -05:00
|
|
|
fn min_by<B: TotalOrd>(&mut self, f: |&A| -> B) -> Option<A> {
|
2013-06-27 06:51:04 +09:00
|
|
|
self.fold(None, |min: Option<(A, B)>, x| {
|
|
|
|
let x_val = f(&x);
|
|
|
|
match min {
|
|
|
|
None => Some((x, x_val)),
|
|
|
|
Some((y, y_val)) => if x_val < y_val {
|
|
|
|
Some((x, x_val))
|
|
|
|
} else {
|
|
|
|
Some((y, y_val))
|
|
|
|
}
|
|
|
|
}
|
2013-09-20 02:08:47 -04:00
|
|
|
}).map(|(x, _)| x)
|
2013-06-27 06:51:04 +09:00
|
|
|
}
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
|
2013-08-09 07:19:23 -07:00
|
|
|
/// A range iterator able to yield elements from both ends
|
|
|
|
pub trait DoubleEndedIterator<A>: Iterator<A> {
|
|
|
|
/// Yield an element from the end of the range, returning `None` if the range is empty.
|
|
|
|
fn next_back(&mut self) -> Option<A>;
|
|
|
|
|
2014-01-23 20:41:57 +01:00
|
|
|
/// Change the direction of the iterator
|
2013-08-09 07:19:23 -07:00
|
|
|
///
|
2014-01-20 11:23:23 +01:00
|
|
|
/// The flipped iterator swaps the ends on an iterator that can already
|
2013-08-09 07:19:23 -07:00
|
|
|
/// be iterated from the front and from the back.
|
|
|
|
///
|
|
|
|
///
|
2014-01-20 11:23:23 +01:00
|
|
|
/// If the iterator also implements RandomAccessIterator, the flipped
|
2013-08-09 07:19:23 -07:00
|
|
|
/// iterator is also random access, with the indices starting at the back
|
|
|
|
/// of the original iterator.
|
|
|
|
///
|
2014-01-20 11:23:23 +01:00
|
|
|
/// Note: Random access with flipped indices still only applies to the first
|
2014-01-25 20:37:51 +13:00
|
|
|
/// `uint::MAX` elements of the original iterator.
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
2014-01-23 20:41:57 +01:00
|
|
|
fn rev(self) -> Rev<Self> {
|
|
|
|
Rev{iter: self}
|
2013-08-09 07:19:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-19 17:52:20 -04:00
|
|
|
/// A double-ended iterator yielding mutable references
|
|
|
|
pub trait MutableDoubleEndedIterator {
|
|
|
|
// FIXME: #5898: should be called `reverse`
|
|
|
|
/// Use an iterator to reverse a container in-place
|
|
|
|
fn reverse_(&mut self);
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: DoubleEndedIterator<&'a mut A>> MutableDoubleEndedIterator for T {
|
2013-08-19 17:52:20 -04:00
|
|
|
// FIXME: #5898: should be called `reverse`
|
|
|
|
/// Use an iterator to reverse a container in-place
|
|
|
|
fn reverse_(&mut self) {
|
|
|
|
loop {
|
|
|
|
match (self.next(), self.next_back()) {
|
2014-02-01 04:35:36 +08:00
|
|
|
(Some(x), Some(y)) => mem::swap(x, y),
|
2013-08-19 17:52:20 -04:00
|
|
|
_ => break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-01 16:46:12 +02:00
|
|
|
|
2013-08-09 07:19:23 -07:00
|
|
|
/// An object implementing random access indexing by `uint`
|
|
|
|
///
|
|
|
|
/// A `RandomAccessIterator` should be either infinite or a `DoubleEndedIterator`.
|
|
|
|
pub trait RandomAccessIterator<A>: Iterator<A> {
|
2014-01-25 20:37:51 +13:00
|
|
|
/// Return the number of indexable elements. At most `std::uint::MAX`
|
2013-08-09 07:19:23 -07:00
|
|
|
/// elements are indexable, even if the iterator represents a longer range.
|
|
|
|
fn indexable(&self) -> uint;
|
|
|
|
|
|
|
|
/// Return an element at an index
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A>;
|
2013-08-09 07:19:23 -07:00
|
|
|
}
|
|
|
|
|
2013-09-01 16:46:12 +02:00
|
|
|
/// An iterator that knows its exact length
|
|
|
|
///
|
|
|
|
/// This trait is a helper for iterators like the vector iterator, so that
|
|
|
|
/// it can support double-ended enumeration.
|
|
|
|
///
|
|
|
|
/// `Iterator::size_hint` *must* return the exact size of the iterator.
|
|
|
|
/// Note that the size must fit in `uint`.
|
|
|
|
pub trait ExactSize<A> : DoubleEndedIterator<A> {
|
2013-08-30 20:00:14 +02:00
|
|
|
/// Return the index of the last element satisfying the specified predicate
|
|
|
|
///
|
|
|
|
/// If no element matches, None is returned.
|
|
|
|
#[inline]
|
2013-11-18 21:15:42 -08:00
|
|
|
fn rposition(&mut self, predicate: |A| -> bool) -> Option<uint> {
|
2013-09-01 16:46:12 +02:00
|
|
|
let (lower, upper) = self.size_hint();
|
|
|
|
assert!(upper == Some(lower));
|
|
|
|
let mut i = lower;
|
2013-08-30 20:00:14 +02:00
|
|
|
loop {
|
|
|
|
match self.next_back() {
|
|
|
|
None => break,
|
|
|
|
Some(x) => {
|
|
|
|
i = match i.checked_sub(&1) {
|
|
|
|
Some(x) => x,
|
2013-10-21 13:08:31 -07:00
|
|
|
None => fail!("rposition: incorrect ExactSize")
|
2013-08-30 20:00:14 +02:00
|
|
|
};
|
|
|
|
if predicate(x) {
|
|
|
|
return Some(i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-30 19:59:49 +02:00
|
|
|
// All adaptors that preserve the size of the wrapped iterator are fine
|
|
|
|
// Adaptors that may overflow in `size_hint` are not, i.e. `Chain`.
|
2013-09-01 16:46:12 +02:00
|
|
|
impl<A, T: ExactSize<A>> ExactSize<(uint, A)> for Enumerate<T> {}
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: ExactSize<A>> ExactSize<A> for Inspect<'a, A, T> {}
|
2014-01-23 20:41:57 +01:00
|
|
|
impl<A, T: ExactSize<A>> ExactSize<A> for Rev<T> {}
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: ExactSize<A>> ExactSize<B> for Map<'a, A, B, T> {}
|
2013-09-01 16:46:12 +02:00
|
|
|
impl<A, B, T: ExactSize<A>, U: ExactSize<B>> ExactSize<(A, B)> for Zip<T, U> {}
|
2013-08-30 19:59:49 +02:00
|
|
|
|
2013-08-09 07:19:23 -07:00
|
|
|
/// An double-ended iterator with the direction inverted
|
|
|
|
#[deriving(Clone)]
|
2014-01-23 20:41:57 +01:00
|
|
|
pub struct Rev<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T
|
2013-08-09 07:19:23 -07:00
|
|
|
}
|
|
|
|
|
2014-01-23 20:41:57 +01:00
|
|
|
impl<A, T: DoubleEndedIterator<A>> Iterator<A> for Rev<T> {
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> { self.iter.next_back() }
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) { self.iter.size_hint() }
|
|
|
|
}
|
|
|
|
|
2014-01-23 20:41:57 +01:00
|
|
|
impl<A, T: DoubleEndedIterator<A>> DoubleEndedIterator<A> for Rev<T> {
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> { self.iter.next() }
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A, T: DoubleEndedIterator<A> + RandomAccessIterator<A>> RandomAccessIterator<A>
|
2014-01-23 20:41:57 +01:00
|
|
|
for Rev<T> {
|
2013-08-09 07:19:23 -07:00
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint { self.iter.indexable() }
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
|
|
|
let amt = self.indexable();
|
|
|
|
self.iter.idx(amt - index - 1)
|
2013-08-09 07:19:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-30 20:59:46 +02:00
|
|
|
/// A mutable reference to an iterator
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct ByRef<'a, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: &'a mut T
|
2013-09-30 20:59:46 +02:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>> Iterator<A> for ByRef<'a, T> {
|
2013-09-30 20:59:46 +02:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> { self.iter.next() }
|
2013-12-11 21:37:45 -05:00
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) { self.iter.size_hint() }
|
2013-09-30 20:59:46 +02:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: DoubleEndedIterator<A>> DoubleEndedIterator<A> for ByRef<'a, T> {
|
2013-09-30 20:59:46 +02:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> { self.iter.next_back() }
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// A trait for iterators over elements which can be added together
|
2013-05-18 00:18:09 +09:00
|
|
|
pub trait AdditiveIterator<A> {
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Iterates over the entire iterator, summing up all the elements
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// use std::iter::AdditiveIterator;
|
|
|
|
///
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-08-09 20:09:47 -07:00
|
|
|
/// let mut it = a.iter().map(|&x| x);
|
2013-05-28 16:35:52 -05:00
|
|
|
/// assert!(it.sum() == 15);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-05-18 00:18:09 +09:00
|
|
|
fn sum(&mut self) -> A;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Add<A, A> + Zero, T: Iterator<A>> AdditiveIterator<A> for T {
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-08-08 11:38:10 -07:00
|
|
|
fn sum(&mut self) -> A {
|
|
|
|
let zero: A = Zero::zero();
|
|
|
|
self.fold(zero, |s, x| s + x)
|
|
|
|
}
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// A trait for iterators over elements whose elements can be multiplied
|
|
|
|
/// together.
|
2013-05-18 00:18:09 +09:00
|
|
|
pub trait MultiplicativeIterator<A> {
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Iterates over the entire iterator, multiplying all the elements
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-22 13:31:23 -08:00
|
|
|
/// use std::iter::{count, MultiplicativeIterator};
|
2013-05-28 16:35:52 -05:00
|
|
|
///
|
|
|
|
/// fn factorial(n: uint) -> uint {
|
2013-08-05 20:02:26 -04:00
|
|
|
/// count(1u, 1).take_while(|&i| i <= n).product()
|
2013-05-28 16:35:52 -05:00
|
|
|
/// }
|
|
|
|
/// assert!(factorial(0) == 1);
|
|
|
|
/// assert!(factorial(1) == 1);
|
|
|
|
/// assert!(factorial(5) == 120);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-05-18 00:18:09 +09:00
|
|
|
fn product(&mut self) -> A;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Mul<A, A> + One, T: Iterator<A>> MultiplicativeIterator<A> for T {
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-08-08 11:38:10 -07:00
|
|
|
fn product(&mut self) -> A {
|
|
|
|
let one: A = One::one();
|
|
|
|
self.fold(one, |p, x| p * x)
|
|
|
|
}
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// A trait for iterators over elements which can be compared to one another.
|
|
|
|
/// The type of each element must ascribe to the `Ord` trait.
|
2013-05-18 00:18:09 +09:00
|
|
|
pub trait OrdIterator<A> {
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Consumes the entire iterator to return the maximum element.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert!(a.iter().max().unwrap() == &5);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-05-18 00:18:09 +09:00
|
|
|
fn max(&mut self) -> Option<A>;
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// Consumes the entire iterator to return the minimum element.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-05-28 16:35:52 -05:00
|
|
|
/// let a = [1, 2, 3, 4, 5];
|
2013-11-17 15:31:50 -03:00
|
|
|
/// assert!(a.iter().min().unwrap() == &1);
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-05-18 00:18:09 +09:00
|
|
|
fn min(&mut self) -> Option<A>;
|
2014-01-24 22:40:54 -08:00
|
|
|
|
2014-02-17 22:53:45 +11:00
|
|
|
/// `min_max` finds the minimum and maximum elements in the iterator.
|
2014-01-24 22:40:54 -08:00
|
|
|
///
|
|
|
|
/// The return type `MinMaxResult` is an enum of three variants:
|
|
|
|
/// - `NoElements` if the iterator is empty.
|
|
|
|
/// - `OneElement(x)` if the iterator has exactly one element.
|
|
|
|
/// - `MinMax(x, y)` is returned otherwise, where `x <= y`. Two values are equal if and only if
|
|
|
|
/// there is more than one element in the iterator and all elements are equal.
|
|
|
|
///
|
|
|
|
/// On an iterator of length `n`, `min_max` does `1.5 * n` comparisons,
|
|
|
|
/// and so faster than calling `min` and `max separately which does `2 * n` comparisons.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use std::iter::{NoElements, OneElement, MinMax};
|
|
|
|
///
|
|
|
|
/// let v: [int, ..0] = [];
|
|
|
|
/// assert_eq!(v.iter().min_max(), NoElements);
|
|
|
|
///
|
|
|
|
/// let v = [1i];
|
|
|
|
/// assert!(v.iter().min_max() == OneElement(&1));
|
|
|
|
///
|
|
|
|
/// let v = [1i, 2, 3, 4, 5];
|
|
|
|
/// assert!(v.iter().min_max() == MinMax(&1, &5));
|
|
|
|
///
|
|
|
|
/// let v = [1i, 2, 3, 4, 5, 6];
|
|
|
|
/// assert!(v.iter().min_max() == MinMax(&1, &6));
|
|
|
|
///
|
|
|
|
/// let v = [1i, 1, 1, 1];
|
|
|
|
/// assert!(v.iter().min_max() == MinMax(&1, &1));
|
|
|
|
/// ```
|
|
|
|
fn min_max(&mut self) -> MinMaxResult<A>;
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
2014-03-08 01:10:32 -05:00
|
|
|
impl<A: TotalOrd, T: Iterator<A>> OrdIterator<A> for T {
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-18 00:18:09 +09:00
|
|
|
fn max(&mut self) -> Option<A> {
|
|
|
|
self.fold(None, |max, x| {
|
|
|
|
match max {
|
|
|
|
None => Some(x),
|
|
|
|
Some(y) => Some(cmp::max(x, y))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2013-06-18 14:45:18 -07:00
|
|
|
#[inline]
|
2013-05-18 00:18:09 +09:00
|
|
|
fn min(&mut self) -> Option<A> {
|
|
|
|
self.fold(None, |min, x| {
|
|
|
|
match min {
|
|
|
|
None => Some(x),
|
|
|
|
Some(y) => Some(cmp::min(x, y))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2014-01-24 22:40:54 -08:00
|
|
|
|
|
|
|
fn min_max(&mut self) -> MinMaxResult<A> {
|
|
|
|
let (mut min, mut max) = match self.next() {
|
|
|
|
None => return NoElements,
|
|
|
|
Some(x) => {
|
|
|
|
match self.next() {
|
|
|
|
None => return OneElement(x),
|
|
|
|
Some(y) => if x < y {(x, y)} else {(y,x)}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
loop {
|
|
|
|
// `first` and `second` are the two next elements we want to look at.
|
|
|
|
// We first compare `first` and `second` (#1). The smaller one is then compared to
|
2014-04-21 00:49:39 -04:00
|
|
|
// current minimum (#2). The larger one is compared to current maximum (#3). This
|
2014-01-24 22:40:54 -08:00
|
|
|
// way we do 3 comparisons for 2 elements.
|
|
|
|
let first = match self.next() {
|
|
|
|
None => break,
|
|
|
|
Some(x) => x
|
|
|
|
};
|
|
|
|
let second = match self.next() {
|
|
|
|
None => {
|
|
|
|
if first < min {
|
|
|
|
min = first;
|
|
|
|
} else if first > max {
|
|
|
|
max = first;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Some(x) => x
|
|
|
|
};
|
|
|
|
if first < second {
|
|
|
|
if first < min {min = first;}
|
|
|
|
if max < second {max = second;}
|
|
|
|
} else {
|
|
|
|
if second < min {min = second;}
|
|
|
|
if max < first {max = first;}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MinMax(min, max)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// `MinMaxResult` is an enum returned by `min_max`. See `OrdIterator::min_max` for more detail.
|
2014-04-30 21:41:03 -07:00
|
|
|
#[deriving(Clone, Eq)]
|
2014-01-24 22:40:54 -08:00
|
|
|
pub enum MinMaxResult<T> {
|
|
|
|
/// Empty iterator
|
|
|
|
NoElements,
|
|
|
|
|
|
|
|
/// Iterator with one element, so the minimum and maximum are the same
|
|
|
|
OneElement(T),
|
|
|
|
|
|
|
|
/// More than one element in the iterator, the first element is not larger than the second
|
|
|
|
MinMax(T, T)
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: Clone> MinMaxResult<T> {
|
|
|
|
/// `into_option` creates an `Option` of type `(T,T)`. The returned `Option` has variant
|
|
|
|
/// `None` if and only if the `MinMaxResult` has variant `NoElements`. Otherwise variant
|
|
|
|
/// `Some(x,y)` is returned where `x <= y`. If `MinMaxResult` has variant `OneElement(x)`,
|
|
|
|
/// performing this operation will make one clone of `x`.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use std::iter::{NoElements, OneElement, MinMax, MinMaxResult};
|
|
|
|
///
|
|
|
|
/// let r: MinMaxResult<int> = NoElements;
|
|
|
|
/// assert_eq!(r.into_option(), None)
|
|
|
|
///
|
|
|
|
/// let r = OneElement(1);
|
|
|
|
/// assert_eq!(r.into_option(), Some((1,1)));
|
|
|
|
///
|
|
|
|
/// let r = MinMax(1,2);
|
|
|
|
/// assert_eq!(r.into_option(), Some((1,2)));
|
|
|
|
/// ```
|
|
|
|
pub fn into_option(self) -> Option<(T,T)> {
|
|
|
|
match self {
|
|
|
|
NoElements => None,
|
|
|
|
OneElement(x) => Some((x.clone(), x)),
|
|
|
|
MinMax(x, y) => Some((x, y))
|
|
|
|
}
|
|
|
|
}
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
2013-12-27 13:53:41 +02:00
|
|
|
/// A trait for iterators that are cloneable.
|
|
|
|
pub trait CloneableIterator {
|
2013-07-18 17:09:08 +02:00
|
|
|
/// Repeats an iterator endlessly
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```rust
|
2013-12-27 13:53:41 +02:00
|
|
|
/// use std::iter::{CloneableIterator, count};
|
2013-12-22 13:31:23 -08:00
|
|
|
///
|
2013-08-09 20:16:07 -07:00
|
|
|
/// let a = count(1,1).take(1);
|
2013-07-18 17:09:08 +02:00
|
|
|
/// let mut cy = a.cycle();
|
|
|
|
/// assert_eq!(cy.next(), Some(1));
|
|
|
|
/// assert_eq!(cy.next(), Some(1));
|
2013-09-23 17:20:36 -07:00
|
|
|
/// ```
|
2013-07-27 23:41:20 +02:00
|
|
|
fn cycle(self) -> Cycle<Self>;
|
2013-07-18 17:09:08 +02:00
|
|
|
}
|
|
|
|
|
2013-12-27 13:53:41 +02:00
|
|
|
impl<A, T: Clone + Iterator<A>> CloneableIterator for T {
|
2013-07-18 17:09:08 +02:00
|
|
|
#[inline]
|
2013-07-27 23:41:20 +02:00
|
|
|
fn cycle(self) -> Cycle<T> {
|
|
|
|
Cycle{orig: self.clone(), iter: self}
|
2013-07-18 17:09:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An iterator that repeats endlessly
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-07-27 23:41:20 +02:00
|
|
|
pub struct Cycle<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
orig: T,
|
|
|
|
iter: T,
|
2013-07-18 17:09:08 +02:00
|
|
|
}
|
|
|
|
|
2013-07-27 23:41:20 +02:00
|
|
|
impl<A, T: Clone + Iterator<A>> Iterator<A> for Cycle<T> {
|
2013-07-18 17:09:08 +02:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
match self.iter.next() {
|
|
|
|
None => { self.iter = self.orig.clone(); self.iter.next() }
|
|
|
|
y => y
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
// the cycle iterator is either empty or infinite
|
|
|
|
match self.orig.size_hint() {
|
|
|
|
sz @ (0, Some(0)) => sz,
|
|
|
|
(0, _) => (0, None),
|
2014-01-25 20:37:51 +13:00
|
|
|
_ => (uint::MAX, None)
|
2013-07-18 17:09:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
impl<A, T: Clone + RandomAccessIterator<A>> RandomAccessIterator<A> for Cycle<T> {
|
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
if self.orig.indexable() > 0 {
|
2014-01-25 20:37:51 +13:00
|
|
|
uint::MAX
|
2013-07-29 17:44:45 +02:00
|
|
|
} else {
|
|
|
|
0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
2013-07-29 17:44:45 +02:00
|
|
|
let liter = self.iter.indexable();
|
|
|
|
let lorig = self.orig.indexable();
|
|
|
|
if lorig == 0 {
|
|
|
|
None
|
|
|
|
} else if index < liter {
|
|
|
|
self.iter.idx(index)
|
|
|
|
} else {
|
|
|
|
self.orig.idx((index - liter) % lorig)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which strings two iterators together
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-07-27 23:41:20 +02:00
|
|
|
pub struct Chain<T, U> {
|
2014-03-27 15:09:47 -07:00
|
|
|
a: T,
|
|
|
|
b: U,
|
2014-05-01 18:06:59 -07:00
|
|
|
flag: bool,
|
2013-04-19 11:29:38 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 23:41:20 +02:00
|
|
|
impl<A, T: Iterator<A>, U: Iterator<A>> Iterator<A> for Chain<T, U> {
|
2013-04-19 11:29:38 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
if self.flag {
|
|
|
|
self.b.next()
|
|
|
|
} else {
|
|
|
|
match self.a.next() {
|
|
|
|
Some(x) => return Some(x),
|
|
|
|
_ => ()
|
|
|
|
}
|
|
|
|
self.flag = true;
|
|
|
|
self.b.next()
|
|
|
|
}
|
|
|
|
}
|
2013-06-21 06:12:01 -04:00
|
|
|
|
|
|
|
#[inline]
|
2013-07-02 18:40:46 -07:00
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
2013-06-21 06:12:01 -04:00
|
|
|
let (a_lower, a_upper) = self.a.size_hint();
|
|
|
|
let (b_lower, b_upper) = self.b.size_hint();
|
|
|
|
|
2013-08-05 20:04:52 -07:00
|
|
|
let lower = a_lower.saturating_add(b_lower);
|
2013-06-21 06:12:01 -04:00
|
|
|
|
|
|
|
let upper = match (a_upper, b_upper) {
|
2013-08-18 21:47:43 -07:00
|
|
|
(Some(x), Some(y)) => x.checked_add(&y),
|
2013-06-21 06:12:01 -04:00
|
|
|
_ => None
|
|
|
|
};
|
|
|
|
|
|
|
|
(lower, upper)
|
|
|
|
}
|
2013-04-19 11:29:38 -04:00
|
|
|
}
|
|
|
|
|
2013-07-18 19:50:51 -04:00
|
|
|
impl<A, T: DoubleEndedIterator<A>, U: DoubleEndedIterator<A>> DoubleEndedIterator<A>
|
2013-07-27 23:41:20 +02:00
|
|
|
for Chain<T, U> {
|
2013-07-18 19:50:51 -04:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> {
|
|
|
|
match self.b.next_back() {
|
|
|
|
Some(x) => Some(x),
|
|
|
|
None => self.a.next_back()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-22 20:11:24 -04:00
|
|
|
impl<A, T: RandomAccessIterator<A>, U: RandomAccessIterator<A>> RandomAccessIterator<A>
|
2013-07-27 23:41:20 +02:00
|
|
|
for Chain<T, U> {
|
2013-07-22 20:11:24 -04:00
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
let (a, b) = (self.a.indexable(), self.b.indexable());
|
2013-08-05 20:04:52 -07:00
|
|
|
a.saturating_add(b)
|
2013-07-22 20:11:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
2013-07-22 20:11:24 -04:00
|
|
|
let len = self.a.indexable();
|
|
|
|
if index < len {
|
|
|
|
self.a.idx(index)
|
|
|
|
} else {
|
|
|
|
self.b.idx(index - len)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which iterates two other iterators simultaneously
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-07-27 23:41:20 +02:00
|
|
|
pub struct Zip<T, U> {
|
2014-03-27 15:09:47 -07:00
|
|
|
a: T,
|
|
|
|
b: U
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 23:41:20 +02:00
|
|
|
impl<A, B, T: Iterator<A>, U: Iterator<B>> Iterator<(A, B)> for Zip<T, U> {
|
2013-04-09 10:54:32 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<(A, B)> {
|
2013-08-03 15:41:29 -07:00
|
|
|
match self.a.next() {
|
|
|
|
None => None,
|
|
|
|
Some(x) => match self.b.next() {
|
|
|
|
None => None,
|
|
|
|
Some(y) => Some((x, y))
|
|
|
|
}
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (a_lower, a_upper) = self.a.size_hint();
|
|
|
|
let (b_lower, b_upper) = self.b.size_hint();
|
|
|
|
|
|
|
|
let lower = cmp::min(a_lower, b_lower);
|
|
|
|
|
|
|
|
let upper = match (a_upper, b_upper) {
|
|
|
|
(Some(x), Some(y)) => Some(cmp::min(x,y)),
|
|
|
|
(Some(x), None) => Some(x),
|
|
|
|
(None, Some(y)) => Some(y),
|
|
|
|
(None, None) => None
|
|
|
|
};
|
|
|
|
|
|
|
|
(lower, upper)
|
|
|
|
}
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
|
2013-09-01 16:46:12 +02:00
|
|
|
impl<A, B, T: ExactSize<A>, U: ExactSize<B>> DoubleEndedIterator<(A, B)>
|
2013-08-30 20:00:07 +02:00
|
|
|
for Zip<T, U> {
|
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<(A, B)> {
|
2013-09-01 16:46:12 +02:00
|
|
|
let (a_sz, a_upper) = self.a.size_hint();
|
|
|
|
let (b_sz, b_upper) = self.b.size_hint();
|
|
|
|
assert!(a_upper == Some(a_sz));
|
|
|
|
assert!(b_upper == Some(b_sz));
|
2013-08-30 20:00:07 +02:00
|
|
|
if a_sz < b_sz {
|
|
|
|
for _ in range(0, b_sz - a_sz) { self.b.next_back(); }
|
|
|
|
} else if a_sz > b_sz {
|
|
|
|
for _ in range(0, a_sz - b_sz) { self.a.next_back(); }
|
|
|
|
}
|
|
|
|
let (a_sz, _) = self.a.size_hint();
|
|
|
|
let (b_sz, _) = self.b.size_hint();
|
|
|
|
assert!(a_sz == b_sz);
|
|
|
|
match (self.a.next_back(), self.b.next_back()) {
|
|
|
|
(Some(x), Some(y)) => Some((x, y)),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
impl<A, B, T: RandomAccessIterator<A>, U: RandomAccessIterator<B>>
|
|
|
|
RandomAccessIterator<(A, B)> for Zip<T, U> {
|
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
cmp::min(self.a.indexable(), self.b.indexable())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<(A, B)> {
|
2013-08-03 15:41:29 -07:00
|
|
|
match self.a.idx(index) {
|
|
|
|
None => None,
|
|
|
|
Some(x) => match self.b.idx(index) {
|
|
|
|
None => None,
|
|
|
|
Some(y) => Some((x, y))
|
|
|
|
}
|
2013-07-29 17:44:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which maps the values of `iter` with `f`
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct Map<'a, A, B, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
2014-04-07 13:30:48 -07:00
|
|
|
f: |A|: 'a -> B
|
2013-04-19 09:18:22 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T> Map<'a, A, B, T> {
|
2013-04-19 09:18:22 -04:00
|
|
|
#[inline]
|
2014-04-21 23:25:18 -07:00
|
|
|
fn do_map(&mut self, elt: Option<A>) -> Option<B> {
|
2013-07-29 17:44:45 +02:00
|
|
|
match elt {
|
2013-04-19 09:18:22 -04:00
|
|
|
Some(a) => Some((self.f)(a)),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
2013-07-29 17:44:45 +02:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: Iterator<A>> Iterator<B> for Map<'a, A, B, T> {
|
2013-07-29 17:44:45 +02:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<B> {
|
|
|
|
let next = self.iter.next();
|
|
|
|
self.do_map(next)
|
|
|
|
}
|
2013-06-21 06:12:01 -04:00
|
|
|
|
|
|
|
#[inline]
|
2013-07-02 18:40:46 -07:00
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
2013-06-21 06:12:01 -04:00
|
|
|
self.iter.size_hint()
|
|
|
|
}
|
2013-04-19 09:18:22 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: DoubleEndedIterator<A>> DoubleEndedIterator<B> for Map<'a, A, B, T> {
|
2013-07-18 19:50:51 -04:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<B> {
|
2013-07-29 17:44:45 +02:00
|
|
|
let next = self.iter.next_back();
|
|
|
|
self.do_map(next)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: RandomAccessIterator<A>> RandomAccessIterator<B> for Map<'a, A, B, T> {
|
2013-07-29 17:44:45 +02:00
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
self.iter.indexable()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<B> {
|
|
|
|
let elt = self.iter.idx(index);
|
|
|
|
self.do_map(elt)
|
2013-07-18 19:50:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which filters the elements of `iter` with `predicate`
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct Filter<'a, A, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
2014-04-07 13:30:48 -07:00
|
|
|
predicate: |&A|: 'a -> bool
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>> Iterator<A> for Filter<'a, A, T> {
|
2013-04-09 10:54:32 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in self.iter {
|
2013-04-09 10:54:32 -04:00
|
|
|
if (self.predicate)(&x) {
|
|
|
|
return Some(x);
|
|
|
|
} else {
|
2013-10-01 14:31:03 -07:00
|
|
|
continue
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
2013-06-21 06:12:01 -04:00
|
|
|
|
|
|
|
#[inline]
|
2013-07-02 18:40:46 -07:00
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
2013-06-21 06:12:01 -04:00
|
|
|
let (_, upper) = self.iter.size_hint();
|
2013-07-02 18:40:46 -07:00
|
|
|
(0, upper) // can't know a lower bound, due to the predicate
|
2013-06-21 06:12:01 -04:00
|
|
|
}
|
2013-04-09 10:54:32 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: DoubleEndedIterator<A>> DoubleEndedIterator<A> for Filter<'a, A, T> {
|
2013-07-18 19:50:51 -04:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> {
|
|
|
|
loop {
|
|
|
|
match self.iter.next_back() {
|
|
|
|
None => return None,
|
|
|
|
Some(x) => {
|
|
|
|
if (self.predicate)(&x) {
|
|
|
|
return Some(x);
|
|
|
|
} else {
|
2013-10-01 14:31:03 -07:00
|
|
|
continue
|
2013-07-18 19:50:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which uses `f` to both filter and map elements from `iter`
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct FilterMap<'a, A, B, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
2014-04-07 13:30:48 -07:00
|
|
|
f: |A|: 'a -> Option<B>
|
2013-05-17 23:00:48 +09:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: Iterator<A>> Iterator<B> for FilterMap<'a, A, B, T> {
|
2013-05-17 23:00:48 +09:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<B> {
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in self.iter {
|
2013-05-18 04:48:22 -04:00
|
|
|
match (self.f)(x) {
|
|
|
|
Some(y) => return Some(y),
|
|
|
|
None => ()
|
2013-05-17 23:00:48 +09:00
|
|
|
}
|
|
|
|
}
|
2013-05-18 04:48:22 -04:00
|
|
|
None
|
2013-05-17 23:00:48 +09:00
|
|
|
}
|
2013-06-21 06:12:01 -04:00
|
|
|
|
|
|
|
#[inline]
|
2013-07-02 18:40:46 -07:00
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
2013-06-21 06:12:01 -04:00
|
|
|
let (_, upper) = self.iter.size_hint();
|
2013-07-02 18:40:46 -07:00
|
|
|
(0, upper) // can't know a lower bound, due to the predicate
|
2013-06-21 06:12:01 -04:00
|
|
|
}
|
2013-05-17 23:00:48 +09:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: DoubleEndedIterator<A>> DoubleEndedIterator<B>
|
|
|
|
for FilterMap<'a, A, B, T> {
|
2013-07-18 19:50:51 -04:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<B> {
|
|
|
|
loop {
|
|
|
|
match self.iter.next_back() {
|
|
|
|
None => return None,
|
|
|
|
Some(x) => {
|
|
|
|
match (self.f)(x) {
|
|
|
|
Some(y) => return Some(y),
|
|
|
|
None => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which yields the current count and the element during iteration
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-07-27 23:41:20 +02:00
|
|
|
pub struct Enumerate<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
count: uint
|
2013-04-18 19:34:07 +10:00
|
|
|
}
|
|
|
|
|
2013-07-27 23:41:20 +02:00
|
|
|
impl<A, T: Iterator<A>> Iterator<(uint, A)> for Enumerate<T> {
|
2013-04-18 19:34:07 +10:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<(uint, A)> {
|
|
|
|
match self.iter.next() {
|
|
|
|
Some(a) => {
|
|
|
|
let ret = Some((self.count, a));
|
|
|
|
self.count += 1;
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
self.iter.size_hint()
|
|
|
|
}
|
2013-04-18 19:34:07 +10:00
|
|
|
}
|
2013-04-18 08:15:40 -04:00
|
|
|
|
2013-09-01 16:46:12 +02:00
|
|
|
impl<A, T: ExactSize<A>> DoubleEndedIterator<(uint, A)> for Enumerate<T> {
|
2013-08-30 19:59:49 +02:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<(uint, A)> {
|
|
|
|
match self.iter.next_back() {
|
|
|
|
Some(a) => {
|
2013-09-01 16:46:12 +02:00
|
|
|
let (lower, upper) = self.iter.size_hint();
|
|
|
|
assert!(upper == Some(lower));
|
|
|
|
Some((self.count + lower, a))
|
2013-08-30 19:59:49 +02:00
|
|
|
}
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
impl<A, T: RandomAccessIterator<A>> RandomAccessIterator<(uint, A)> for Enumerate<T> {
|
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
self.iter.indexable()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<(uint, A)> {
|
2013-07-29 17:44:45 +02:00
|
|
|
match self.iter.idx(index) {
|
|
|
|
Some(a) => Some((self.count + index, a)),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-08 20:46:36 +01:00
|
|
|
/// An iterator with a `peek()` that returns an optional reference to the next element.
|
|
|
|
pub struct Peekable<A, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
peeked: Option<A>,
|
2013-08-08 20:46:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<A, T: Iterator<A>> Iterator<A> for Peekable<A, T> {
|
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
if self.peeked.is_some() { self.peeked.take() }
|
|
|
|
else { self.iter.next() }
|
|
|
|
}
|
2013-08-18 21:47:43 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (lo, hi) = self.iter.size_hint();
|
|
|
|
if self.peeked.is_some() {
|
|
|
|
let lo = lo.saturating_add(1);
|
|
|
|
let hi = match hi {
|
|
|
|
Some(x) => x.checked_add(&1),
|
|
|
|
None => None
|
|
|
|
};
|
|
|
|
(lo, hi)
|
|
|
|
} else {
|
|
|
|
(lo, hi)
|
|
|
|
}
|
|
|
|
}
|
2013-08-08 20:46:36 +01:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>> Peekable<A, T> {
|
2013-08-08 20:46:36 +01:00
|
|
|
/// Return a reference to the next element of the iterator with out advancing it,
|
|
|
|
/// or None if the iterator is exhausted.
|
|
|
|
#[inline]
|
2013-12-09 23:16:18 -08:00
|
|
|
pub fn peek(&'a mut self) -> Option<&'a A> {
|
2013-08-18 21:47:43 -07:00
|
|
|
if self.peeked.is_none() {
|
|
|
|
self.peeked = self.iter.next();
|
|
|
|
}
|
2013-08-08 20:46:36 +01:00
|
|
|
match self.peeked {
|
|
|
|
Some(ref value) => Some(value),
|
2013-08-18 21:47:43 -07:00
|
|
|
None => None,
|
2013-08-08 20:46:36 +01:00
|
|
|
}
|
|
|
|
}
|
2014-01-11 14:13:06 +04:00
|
|
|
|
|
|
|
/// Check whether peekable iterator is empty or not.
|
|
|
|
#[inline]
|
2014-01-14 18:38:06 +04:00
|
|
|
pub fn is_empty(&mut self) -> bool {
|
2014-01-25 02:44:06 -05:00
|
|
|
self.peek().is_none()
|
2014-01-11 14:13:06 +04:00
|
|
|
}
|
2013-08-08 20:46:36 +01:00
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which rejects elements while `predicate` is true
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct SkipWhile<'a, A, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
flag: bool,
|
2014-04-07 13:30:48 -07:00
|
|
|
predicate: |&A|: 'a -> bool
|
2013-04-18 08:15:40 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>> Iterator<A> for SkipWhile<'a, A, T> {
|
2013-04-18 08:15:40 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
let mut next = self.iter.next();
|
|
|
|
if self.flag {
|
|
|
|
next
|
|
|
|
} else {
|
|
|
|
loop {
|
|
|
|
match next {
|
|
|
|
Some(x) => {
|
|
|
|
if (self.predicate)(&x) {
|
|
|
|
next = self.iter.next();
|
2013-10-01 14:31:03 -07:00
|
|
|
continue
|
2013-04-18 08:15:40 -04:00
|
|
|
} else {
|
|
|
|
self.flag = true;
|
|
|
|
return Some(x)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => return None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (_, upper) = self.iter.size_hint();
|
|
|
|
(0, upper) // can't know a lower bound, due to the predicate
|
|
|
|
}
|
2013-04-18 08:15:40 -04:00
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which only accepts elements while `predicate` is true
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct TakeWhile<'a, A, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
flag: bool,
|
2014-04-07 13:30:48 -07:00
|
|
|
predicate: |&A|: 'a -> bool
|
2013-04-18 08:15:40 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>> Iterator<A> for TakeWhile<'a, A, T> {
|
2013-04-18 08:15:40 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
if self.flag {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
match self.iter.next() {
|
|
|
|
Some(x) => {
|
|
|
|
if (self.predicate)(&x) {
|
|
|
|
Some(x)
|
|
|
|
} else {
|
|
|
|
self.flag = true;
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (_, upper) = self.iter.size_hint();
|
|
|
|
(0, upper) // can't know a lower bound, due to the predicate
|
|
|
|
}
|
2013-04-18 08:15:40 -04:00
|
|
|
}
|
2013-04-19 06:06:33 -04:00
|
|
|
|
2013-06-06 16:39:25 +10:00
|
|
|
/// An iterator which skips over `n` elements of `iter`.
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-07-27 23:41:20 +02:00
|
|
|
pub struct Skip<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
n: uint
|
2013-04-19 06:06:33 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 23:41:20 +02:00
|
|
|
impl<A, T: Iterator<A>> Iterator<A> for Skip<T> {
|
2013-04-19 06:06:33 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
let mut next = self.iter.next();
|
|
|
|
if self.n == 0 {
|
|
|
|
next
|
|
|
|
} else {
|
2013-08-01 04:18:19 +02:00
|
|
|
let mut n = self.n;
|
|
|
|
while n > 0 {
|
|
|
|
n -= 1;
|
2013-04-19 06:06:33 -04:00
|
|
|
match next {
|
|
|
|
Some(_) => {
|
|
|
|
next = self.iter.next();
|
2013-10-01 14:31:03 -07:00
|
|
|
continue
|
2013-04-19 06:06:33 -04:00
|
|
|
}
|
|
|
|
None => {
|
|
|
|
self.n = 0;
|
|
|
|
return None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
self.n = 0;
|
|
|
|
next
|
|
|
|
}
|
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (lower, upper) = self.iter.size_hint();
|
|
|
|
|
2013-08-05 20:04:52 -07:00
|
|
|
let lower = lower.saturating_sub(self.n);
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
let upper = match upper {
|
2013-08-05 20:04:52 -07:00
|
|
|
Some(x) => Some(x.saturating_sub(self.n)),
|
2013-07-02 20:59:26 -07:00
|
|
|
None => None
|
|
|
|
};
|
|
|
|
|
|
|
|
(lower, upper)
|
|
|
|
}
|
2013-04-19 06:06:33 -04:00
|
|
|
}
|
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
impl<A, T: RandomAccessIterator<A>> RandomAccessIterator<A> for Skip<T> {
|
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
2013-08-05 20:04:52 -07:00
|
|
|
self.iter.indexable().saturating_sub(self.n)
|
2013-07-29 17:44:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
2013-07-29 17:44:45 +02:00
|
|
|
if index >= self.indexable() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
self.iter.idx(index + self.n)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which only iterates over the first `n` iterations of `iter`.
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-07-27 23:41:20 +02:00
|
|
|
pub struct Take<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
n: uint
|
2013-04-19 06:06:33 -04:00
|
|
|
}
|
|
|
|
|
2013-07-27 23:41:20 +02:00
|
|
|
impl<A, T: Iterator<A>> Iterator<A> for Take<T> {
|
2013-04-19 06:06:33 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
if self.n != 0 {
|
|
|
|
self.n -= 1;
|
2013-08-05 19:23:29 -04:00
|
|
|
self.iter.next()
|
2013-04-19 06:06:33 -04:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (lower, upper) = self.iter.size_hint();
|
|
|
|
|
|
|
|
let lower = cmp::min(lower, self.n);
|
|
|
|
|
|
|
|
let upper = match upper {
|
|
|
|
Some(x) if x < self.n => Some(x),
|
|
|
|
_ => Some(self.n)
|
|
|
|
};
|
|
|
|
|
|
|
|
(lower, upper)
|
|
|
|
}
|
2013-04-19 06:06:33 -04:00
|
|
|
}
|
2013-04-19 07:30:22 -04:00
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
impl<A, T: RandomAccessIterator<A>> RandomAccessIterator<A> for Take<T> {
|
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
cmp::min(self.iter.indexable(), self.n)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
2013-07-29 17:44:45 +02:00
|
|
|
if index >= self.n {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
self.iter.idx(index)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator to maintain state while iterating another iterator
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct Scan<'a, A, B, T, St> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
2014-04-07 13:30:48 -07:00
|
|
|
f: |&mut St, A|: 'a -> Option<B>,
|
2013-05-28 16:35:52 -05:00
|
|
|
|
|
|
|
/// The current internal state to be passed to the closure next.
|
2014-03-27 15:09:47 -07:00
|
|
|
pub state: St,
|
2013-04-24 19:54:13 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, B, T: Iterator<A>, St> Iterator<B> for Scan<'a, A, B, T, St> {
|
2013-04-24 19:54:13 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<B> {
|
2013-09-11 12:52:17 -07:00
|
|
|
self.iter.next().and_then(|a| (self.f)(&mut self.state, a))
|
2013-04-24 19:54:13 -04:00
|
|
|
}
|
2013-07-02 20:59:26 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (_, upper) = self.iter.size_hint();
|
|
|
|
(0, upper) // can't know a lower bound, due to the scan function
|
|
|
|
}
|
2013-04-24 19:54:13 -04:00
|
|
|
}
|
|
|
|
|
2013-06-24 02:34:38 +02:00
|
|
|
/// An iterator that maps each element to an iterator,
|
|
|
|
/// and yields the elements of the produced iterators
|
|
|
|
///
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct FlatMap<'a, A, T, U> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
2014-04-07 13:30:48 -07:00
|
|
|
f: |A|: 'a -> U,
|
2014-03-27 15:09:47 -07:00
|
|
|
frontiter: Option<U>,
|
|
|
|
backiter: Option<U>,
|
2013-06-24 02:34:38 +02:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>, B, U: Iterator<B>> Iterator<B> for FlatMap<'a, A, T, U> {
|
2013-06-24 02:34:38 +02:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<B> {
|
|
|
|
loop {
|
2013-08-03 12:45:23 -04:00
|
|
|
for inner in self.frontiter.mut_iter() {
|
|
|
|
for x in *inner {
|
2013-06-24 02:34:38 +02:00
|
|
|
return Some(x)
|
|
|
|
}
|
|
|
|
}
|
2013-09-20 02:08:47 -04:00
|
|
|
match self.iter.next().map(|x| (self.f)(x)) {
|
|
|
|
None => return self.backiter.as_mut().and_then(|it| it.next()),
|
2013-07-20 03:11:28 +02:00
|
|
|
next => self.frontiter = next,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-07-20 17:10:00 +02:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
2013-12-06 19:51:10 +01:00
|
|
|
let (flo, fhi) = self.frontiter.as_ref().map_or((0, Some(0)), |it| it.size_hint());
|
|
|
|
let (blo, bhi) = self.backiter.as_ref().map_or((0, Some(0)), |it| it.size_hint());
|
2013-08-05 20:04:52 -07:00
|
|
|
let lo = flo.saturating_add(blo);
|
2013-07-20 17:10:00 +02:00
|
|
|
match (self.iter.size_hint(), fhi, bhi) {
|
2013-08-18 21:47:43 -07:00
|
|
|
((0, Some(0)), Some(a), Some(b)) => (lo, a.checked_add(&b)),
|
2013-08-05 20:04:52 -07:00
|
|
|
_ => (lo, None)
|
2013-07-20 17:10:00 +02:00
|
|
|
}
|
|
|
|
}
|
2013-07-20 03:11:28 +02:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a,
|
2013-07-20 03:11:28 +02:00
|
|
|
A, T: DoubleEndedIterator<A>,
|
|
|
|
B, U: DoubleEndedIterator<B>> DoubleEndedIterator<B>
|
2013-12-09 23:16:18 -08:00
|
|
|
for FlatMap<'a, A, T, U> {
|
2013-07-20 03:11:28 +02:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<B> {
|
|
|
|
loop {
|
2013-08-03 12:45:23 -04:00
|
|
|
for inner in self.backiter.mut_iter() {
|
2013-07-20 03:11:28 +02:00
|
|
|
match inner.next_back() {
|
|
|
|
None => (),
|
|
|
|
y => return y
|
|
|
|
}
|
|
|
|
}
|
2013-09-20 02:08:47 -04:00
|
|
|
match self.iter.next_back().map(|x| (self.f)(x)) {
|
|
|
|
None => return self.frontiter.as_mut().and_then(|it| it.next_back()),
|
2013-07-20 03:11:28 +02:00
|
|
|
next => self.backiter = next,
|
2013-06-24 02:34:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-03 13:51:49 -07:00
|
|
|
/// An iterator that yields `None` forever after the underlying iterator
|
|
|
|
/// yields `None` once.
|
2014-03-05 01:19:14 -05:00
|
|
|
#[deriving(Clone)]
|
2013-08-03 13:51:49 -07:00
|
|
|
pub struct Fuse<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
|
|
|
done: bool
|
2013-08-03 13:51:49 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<A, T: Iterator<A>> Iterator<A> for Fuse<T> {
|
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
if self.done {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
match self.iter.next() {
|
|
|
|
None => {
|
|
|
|
self.done = true;
|
|
|
|
None
|
|
|
|
}
|
|
|
|
x => x
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
if self.done {
|
|
|
|
(0, Some(0))
|
|
|
|
} else {
|
|
|
|
self.iter.size_hint()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A, T: DoubleEndedIterator<A>> DoubleEndedIterator<A> for Fuse<T> {
|
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> {
|
|
|
|
if self.done {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
match self.iter.next_back() {
|
|
|
|
None => {
|
|
|
|
self.done = true;
|
|
|
|
None
|
|
|
|
}
|
|
|
|
x => x
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allow RandomAccessIterators to be fused without affecting random-access behavior
|
|
|
|
impl<A, T: RandomAccessIterator<A>> RandomAccessIterator<A> for Fuse<T> {
|
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
self.iter.indexable()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
2013-08-03 13:51:49 -07:00
|
|
|
self.iter.idx(index)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> Fuse<T> {
|
|
|
|
/// Resets the fuse such that the next call to .next() or .next_back() will
|
2013-12-15 16:26:09 +11:00
|
|
|
/// call the underlying iterator again even if it previously returned None.
|
2013-08-03 13:51:49 -07:00
|
|
|
#[inline]
|
2013-12-08 02:55:28 -05:00
|
|
|
pub fn reset_fuse(&mut self) {
|
2013-08-03 13:51:49 -07:00
|
|
|
self.done = false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-05 13:00:11 -04:00
|
|
|
/// An iterator that calls a function with a reference to each
|
|
|
|
/// element before yielding it.
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct Inspect<'a, A, T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
iter: T,
|
2014-04-07 13:30:48 -07:00
|
|
|
f: |&A|: 'a
|
2013-07-05 13:00:11 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T> Inspect<'a, A, T> {
|
2013-07-05 13:00:11 -04:00
|
|
|
#[inline]
|
2014-04-21 23:25:18 -07:00
|
|
|
fn do_inspect(&mut self, elt: Option<A>) -> Option<A> {
|
2013-07-29 17:44:45 +02:00
|
|
|
match elt {
|
2013-07-05 13:00:11 -04:00
|
|
|
Some(ref a) => (self.f)(a),
|
|
|
|
None => ()
|
|
|
|
}
|
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
elt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: Iterator<A>> Iterator<A> for Inspect<'a, A, T> {
|
2013-07-29 17:44:45 +02:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
let next = self.iter.next();
|
2013-08-11 01:55:34 +02:00
|
|
|
self.do_inspect(next)
|
2013-07-05 13:00:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
self.iter.size_hint()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: DoubleEndedIterator<A>> DoubleEndedIterator<A>
|
|
|
|
for Inspect<'a, A, T> {
|
2013-07-18 19:50:51 -04:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> {
|
|
|
|
let next = self.iter.next_back();
|
2013-08-11 01:55:34 +02:00
|
|
|
self.do_inspect(next)
|
2013-07-29 17:44:45 +02:00
|
|
|
}
|
|
|
|
}
|
2013-07-18 19:50:51 -04:00
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, T: RandomAccessIterator<A>> RandomAccessIterator<A>
|
|
|
|
for Inspect<'a, A, T> {
|
2013-07-29 17:44:45 +02:00
|
|
|
#[inline]
|
|
|
|
fn indexable(&self) -> uint {
|
|
|
|
self.iter.indexable()
|
|
|
|
}
|
2013-07-18 19:50:51 -04:00
|
|
|
|
2013-07-29 17:44:45 +02:00
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, index: uint) -> Option<A> {
|
|
|
|
let element = self.iter.idx(index);
|
|
|
|
self.do_inspect(element)
|
2013-07-18 19:50:51 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An iterator which just modifies the contained state throughout iteration.
|
2013-12-09 23:16:18 -08:00
|
|
|
pub struct Unfold<'a, A, St> {
|
2014-04-07 13:30:48 -07:00
|
|
|
f: |&mut St|: 'a -> Option<A>,
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Internal state that will be yielded on the next iteration
|
2014-03-27 15:09:47 -07:00
|
|
|
pub state: St,
|
2013-04-20 00:07:07 +10:00
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, St> Unfold<'a, A, St> {
|
2013-05-28 16:35:52 -05:00
|
|
|
/// Creates a new iterator with the specified closure as the "iterator
|
|
|
|
/// function" and an initial state to eventually pass to the iterator
|
2013-04-20 00:07:07 +10:00
|
|
|
#[inline]
|
2014-04-07 13:30:48 -07:00
|
|
|
pub fn new<'a>(initial_state: St, f: |&mut St|: 'a -> Option<A>)
|
2013-11-19 17:36:32 -08:00
|
|
|
-> Unfold<'a, A, St> {
|
2013-09-08 10:52:19 +10:00
|
|
|
Unfold {
|
2013-04-20 00:07:07 +10:00
|
|
|
f: f,
|
|
|
|
state: initial_state
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 23:16:18 -08:00
|
|
|
impl<'a, A, St> Iterator<A> for Unfold<'a, A, St> {
|
2013-04-20 00:07:07 +10:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
(self.f)(&mut self.state)
|
|
|
|
}
|
2013-08-18 21:47:43 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
// no possible known bounds at this point
|
|
|
|
(0, None)
|
|
|
|
}
|
2013-04-20 00:07:07 +10:00
|
|
|
}
|
|
|
|
|
2013-05-28 16:35:52 -05:00
|
|
|
/// An infinite iterator starting at `start` and advancing by `step` with each
|
|
|
|
/// iteration
|
2013-07-18 17:31:33 +02:00
|
|
|
#[deriving(Clone)]
|
2013-04-24 19:54:13 -04:00
|
|
|
pub struct Counter<A> {
|
2013-05-28 16:35:52 -05:00
|
|
|
/// The current state the counter is at (next value to be yielded)
|
2014-03-27 15:09:47 -07:00
|
|
|
state: A,
|
2013-05-28 16:35:52 -05:00
|
|
|
/// The amount that this iterator is stepping by
|
2014-03-27 15:09:47 -07:00
|
|
|
step: A,
|
2013-04-20 13:32:27 +10:00
|
|
|
}
|
|
|
|
|
2013-08-05 20:02:26 -04:00
|
|
|
/// Creates a new counter with the specified start/step
|
|
|
|
#[inline]
|
|
|
|
pub fn count<A>(start: A, step: A) -> Counter<A> {
|
|
|
|
Counter{state: start, step: step}
|
2013-04-24 19:54:13 -04:00
|
|
|
}
|
|
|
|
|
2013-09-14 16:37:52 -04:00
|
|
|
impl<A: Add<A, A> + Clone> Iterator<A> for Counter<A> {
|
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
let result = self.state.clone();
|
|
|
|
self.state = self.state + self.step;
|
|
|
|
Some(result)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
2014-01-25 20:37:51 +13:00
|
|
|
(uint::MAX, None) // Too bad we can't specify an infinite lower bound
|
2013-09-14 16:37:52 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-14 16:33:19 -04:00
|
|
|
/// An iterator over the range [start, stop)
|
2014-03-05 01:19:14 -05:00
|
|
|
#[deriving(Clone)]
|
2013-08-01 18:35:46 -04:00
|
|
|
pub struct Range<A> {
|
2014-03-27 15:09:47 -07:00
|
|
|
state: A,
|
|
|
|
stop: A,
|
|
|
|
one: A
|
2013-08-01 18:35:46 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return an iterator over the range [start, stop)
|
|
|
|
#[inline]
|
|
|
|
pub fn range<A: Add<A, A> + Ord + Clone + One>(start: A, stop: A) -> Range<A> {
|
|
|
|
Range{state: start, stop: stop, one: One::one()}
|
|
|
|
}
|
|
|
|
|
2013-11-11 05:06:26 -05:00
|
|
|
// FIXME: #10414: Unfortunate type bound
|
|
|
|
impl<A: Add<A, A> + Ord + Clone + ToPrimitive> Iterator<A> for Range<A> {
|
2013-08-01 18:35:46 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
if self.state < self.stop {
|
|
|
|
let result = self.state.clone();
|
|
|
|
self.state = self.state + self.one;
|
|
|
|
Some(result)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
2013-08-18 21:47:43 -07:00
|
|
|
|
2013-11-11 05:06:26 -05:00
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
// This first checks if the elements are representable as i64. If they aren't, try u64 (to
|
|
|
|
// handle cases like range(huge, huger)). We don't use uint/int because the difference of
|
|
|
|
// the i64/u64 might lie within their range.
|
|
|
|
let bound = match self.state.to_i64() {
|
|
|
|
Some(a) => {
|
|
|
|
let sz = self.stop.to_i64().map(|b| b.checked_sub(&a));
|
|
|
|
match sz {
|
|
|
|
Some(Some(bound)) => bound.to_uint(),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => match self.state.to_u64() {
|
|
|
|
Some(a) => {
|
|
|
|
let sz = self.stop.to_u64().map(|b| b.checked_sub(&a));
|
|
|
|
match sz {
|
|
|
|
Some(Some(bound)) => bound.to_uint(),
|
|
|
|
_ => None
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => None
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
match bound {
|
|
|
|
Some(b) => (b, Some(b)),
|
|
|
|
// Standard fallback for unbounded/unrepresentable bounds
|
|
|
|
None => (0, None)
|
|
|
|
}
|
|
|
|
}
|
2013-08-01 18:35:46 -04:00
|
|
|
}
|
|
|
|
|
2014-02-17 07:20:01 +11:00
|
|
|
/// `Int` is required to ensure the range will be the same regardless of
|
2013-09-14 16:34:32 -04:00
|
|
|
/// the direction it is consumed.
|
2014-02-17 07:20:01 +11:00
|
|
|
impl<A: Int + Ord + Clone + ToPrimitive> DoubleEndedIterator<A> for Range<A> {
|
2013-08-06 22:34:22 -07:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> {
|
|
|
|
if self.stop > self.state {
|
|
|
|
self.stop = self.stop - self.one;
|
|
|
|
Some(self.stop.clone())
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-14 16:33:19 -04:00
|
|
|
/// An iterator over the range [start, stop]
|
2014-03-05 01:19:14 -05:00
|
|
|
#[deriving(Clone)]
|
2013-08-17 18:41:53 -04:00
|
|
|
pub struct RangeInclusive<A> {
|
2014-03-27 15:09:47 -07:00
|
|
|
range: Range<A>,
|
|
|
|
done: bool,
|
2013-08-17 18:41:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return an iterator over the range [start, stop]
|
|
|
|
#[inline]
|
2013-11-11 05:06:26 -05:00
|
|
|
pub fn range_inclusive<A: Add<A, A> + Ord + Clone + One + ToPrimitive>(start: A, stop: A)
|
|
|
|
-> RangeInclusive<A> {
|
2013-08-17 18:41:53 -04:00
|
|
|
RangeInclusive{range: range(start, stop), done: false}
|
|
|
|
}
|
|
|
|
|
2014-02-24 08:11:00 -05:00
|
|
|
impl<A: Add<A, A> + Ord + Clone + ToPrimitive> Iterator<A> for RangeInclusive<A> {
|
2013-08-17 18:41:53 -04:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
|
|
|
match self.range.next() {
|
|
|
|
Some(x) => Some(x),
|
|
|
|
None => {
|
2013-09-15 00:27:52 -04:00
|
|
|
if !self.done && self.range.state == self.range.stop {
|
2013-08-17 18:41:53 -04:00
|
|
|
self.done = true;
|
|
|
|
Some(self.range.stop.clone())
|
2013-09-15 00:27:52 -04:00
|
|
|
} else {
|
|
|
|
None
|
2013-08-17 18:41:53 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-21 13:58:00 -07:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
let (lo, hi) = self.range.size_hint();
|
|
|
|
if self.done {
|
|
|
|
(lo, hi)
|
|
|
|
} else {
|
|
|
|
let lo = lo.saturating_add(1);
|
|
|
|
let hi = match hi {
|
|
|
|
Some(x) => x.checked_add(&1),
|
|
|
|
None => None
|
|
|
|
};
|
|
|
|
(lo, hi)
|
|
|
|
}
|
|
|
|
}
|
2013-08-17 18:41:53 -04:00
|
|
|
}
|
|
|
|
|
2014-02-17 07:20:01 +11:00
|
|
|
impl<A: Sub<A, A> + Int + Ord + Clone + ToPrimitive> DoubleEndedIterator<A>
|
2013-11-11 05:06:26 -05:00
|
|
|
for RangeInclusive<A> {
|
2013-08-17 18:41:53 -04:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> {
|
|
|
|
if self.range.stop > self.range.state {
|
|
|
|
let result = self.range.stop.clone();
|
|
|
|
self.range.stop = self.range.stop - self.range.one;
|
|
|
|
Some(result)
|
2013-09-15 00:39:34 -04:00
|
|
|
} else if !self.done && self.range.state == self.range.stop {
|
2013-08-17 18:41:53 -04:00
|
|
|
self.done = true;
|
|
|
|
Some(self.range.stop.clone())
|
2013-09-15 00:39:34 -04:00
|
|
|
} else {
|
|
|
|
None
|
2013-08-17 18:41:53 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-14 16:47:21 -04:00
|
|
|
/// An iterator over the range [start, stop) by `step`. It handles overflow by stopping.
|
2014-03-05 01:19:14 -05:00
|
|
|
#[deriving(Clone)]
|
2013-09-14 16:47:21 -04:00
|
|
|
pub struct RangeStep<A> {
|
2014-03-27 15:09:47 -07:00
|
|
|
state: A,
|
|
|
|
stop: A,
|
|
|
|
step: A,
|
|
|
|
rev: bool,
|
2013-09-14 16:47:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return an iterator over the range [start, stop) by `step`. It handles overflow by stopping.
|
|
|
|
#[inline]
|
|
|
|
pub fn range_step<A: CheckedAdd + Ord + Clone + Zero>(start: A, stop: A, step: A) -> RangeStep<A> {
|
|
|
|
let rev = step < Zero::zero();
|
|
|
|
RangeStep{state: start, stop: stop, step: step, rev: rev}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: CheckedAdd + Ord + Clone> Iterator<A> for RangeStep<A> {
|
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
2013-09-15 00:11:50 -04:00
|
|
|
if (self.rev && self.state > self.stop) || (!self.rev && self.state < self.stop) {
|
2013-09-14 16:47:21 -04:00
|
|
|
let result = self.state.clone();
|
|
|
|
match self.state.checked_add(&self.step) {
|
|
|
|
Some(x) => self.state = x,
|
|
|
|
None => self.state = self.stop.clone()
|
|
|
|
}
|
|
|
|
Some(result)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An iterator over the range [start, stop] by `step`. It handles overflow by stopping.
|
2014-03-05 01:19:14 -05:00
|
|
|
#[deriving(Clone)]
|
2013-09-14 16:47:21 -04:00
|
|
|
pub struct RangeStepInclusive<A> {
|
2014-03-27 15:09:47 -07:00
|
|
|
state: A,
|
|
|
|
stop: A,
|
|
|
|
step: A,
|
|
|
|
rev: bool,
|
|
|
|
done: bool,
|
2013-09-14 16:47:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return an iterator over the range [start, stop] by `step`. It handles overflow by stopping.
|
|
|
|
#[inline]
|
|
|
|
pub fn range_step_inclusive<A: CheckedAdd + Ord + Clone + Zero>(start: A, stop: A,
|
|
|
|
step: A) -> RangeStepInclusive<A> {
|
|
|
|
let rev = step < Zero::zero();
|
|
|
|
RangeStepInclusive{state: start, stop: stop, step: step, rev: rev, done: false}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: CheckedAdd + Ord + Clone + Eq> Iterator<A> for RangeStepInclusive<A> {
|
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> {
|
2013-09-15 00:11:50 -04:00
|
|
|
if !self.done && ((self.rev && self.state >= self.stop) ||
|
|
|
|
(!self.rev && self.state <= self.stop)) {
|
|
|
|
let result = self.state.clone();
|
|
|
|
match self.state.checked_add(&self.step) {
|
|
|
|
Some(x) => self.state = x,
|
|
|
|
None => self.done = true
|
2013-09-14 16:47:21 -04:00
|
|
|
}
|
2013-09-15 00:11:50 -04:00
|
|
|
Some(result)
|
2013-09-14 16:47:21 -04:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-03 21:34:00 +02:00
|
|
|
/// An iterator that repeats an element endlessly
|
2014-03-05 01:19:14 -05:00
|
|
|
#[deriving(Clone)]
|
2013-08-03 21:34:00 +02:00
|
|
|
pub struct Repeat<A> {
|
2014-03-27 15:09:47 -07:00
|
|
|
element: A
|
2013-08-03 21:34:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Clone> Repeat<A> {
|
2013-08-18 08:28:04 +10:00
|
|
|
/// Create a new `Repeat` that endlessly repeats the element `elt`.
|
2013-08-03 21:34:00 +02:00
|
|
|
#[inline]
|
|
|
|
pub fn new(elt: A) -> Repeat<A> {
|
|
|
|
Repeat{element: elt}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Clone> Iterator<A> for Repeat<A> {
|
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<A> { self.idx(0) }
|
|
|
|
#[inline]
|
2014-01-25 20:37:51 +13:00
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) { (uint::MAX, None) }
|
2013-08-03 21:34:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Clone> DoubleEndedIterator<A> for Repeat<A> {
|
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<A> { self.idx(0) }
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<A: Clone> RandomAccessIterator<A> for Repeat<A> {
|
|
|
|
#[inline]
|
2014-01-25 20:37:51 +13:00
|
|
|
fn indexable(&self) -> uint { uint::MAX }
|
2013-08-03 21:34:00 +02:00
|
|
|
#[inline]
|
2014-04-21 22:15:42 -07:00
|
|
|
fn idx(&mut self, _: uint) -> Option<A> { Some(self.element.clone()) }
|
2013-08-03 21:34:00 +02:00
|
|
|
}
|
|
|
|
|
2013-08-08 22:07:21 +02:00
|
|
|
/// Functions for lexicographical ordering of sequences.
|
|
|
|
///
|
|
|
|
/// Lexicographical ordering through `<`, `<=`, `>=`, `>` requires
|
|
|
|
/// that the elements implement both `Eq` and `Ord`.
|
|
|
|
///
|
|
|
|
/// If two sequences are equal up until the point where one ends,
|
|
|
|
/// the shorter sequence compares less.
|
|
|
|
pub mod order {
|
|
|
|
use cmp;
|
|
|
|
use cmp::{TotalEq, TotalOrd, Ord, Eq};
|
|
|
|
use option::{Some, None};
|
|
|
|
use super::Iterator;
|
|
|
|
|
2014-03-23 22:54:42 +11:00
|
|
|
/// Compare `a` and `b` for equality using `TotalEq`
|
2013-08-08 22:07:21 +02:00
|
|
|
pub fn equals<A: TotalEq, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return true,
|
|
|
|
(None, _) | (_, None) => return false,
|
2014-03-23 22:54:42 +11:00
|
|
|
(Some(x), Some(y)) => if x != y { return false },
|
2013-08-08 22:07:21 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Order `a` and `b` lexicographically using `TotalOrd`
|
|
|
|
pub fn cmp<A: TotalOrd, T: Iterator<A>>(mut a: T, mut b: T) -> cmp::Ordering {
|
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return cmp::Equal,
|
|
|
|
(None, _ ) => return cmp::Less,
|
|
|
|
(_ , None) => return cmp::Greater,
|
|
|
|
(Some(x), Some(y)) => match x.cmp(&y) {
|
|
|
|
cmp::Equal => (),
|
|
|
|
non_eq => return non_eq,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Compare `a` and `b` for equality (Using partial equality, `Eq`)
|
|
|
|
pub fn eq<A: Eq, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return true,
|
|
|
|
(None, _) | (_, None) => return false,
|
|
|
|
(Some(x), Some(y)) => if !x.eq(&y) { return false },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Compare `a` and `b` for nonequality (Using partial equality, `Eq`)
|
|
|
|
pub fn ne<A: Eq, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return false,
|
|
|
|
(None, _) | (_, None) => return true,
|
|
|
|
(Some(x), Some(y)) => if x.ne(&y) { return true },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return `a` < `b` lexicographically (Using partial order, `Ord`)
|
2014-02-24 08:11:00 -05:00
|
|
|
pub fn lt<A: Ord, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
2013-08-08 22:07:21 +02:00
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return false,
|
|
|
|
(None, _ ) => return true,
|
|
|
|
(_ , None) => return false,
|
|
|
|
(Some(x), Some(y)) => if x.ne(&y) { return x.lt(&y) },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return `a` <= `b` lexicographically (Using partial order, `Ord`)
|
2014-02-24 08:11:00 -05:00
|
|
|
pub fn le<A: Ord, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
2013-08-08 22:07:21 +02:00
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return true,
|
|
|
|
(None, _ ) => return true,
|
|
|
|
(_ , None) => return false,
|
|
|
|
(Some(x), Some(y)) => if x.ne(&y) { return x.le(&y) },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return `a` > `b` lexicographically (Using partial order, `Ord`)
|
2014-02-24 08:11:00 -05:00
|
|
|
pub fn gt<A: Ord, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
2013-08-08 22:07:21 +02:00
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return false,
|
|
|
|
(None, _ ) => return false,
|
|
|
|
(_ , None) => return true,
|
|
|
|
(Some(x), Some(y)) => if x.ne(&y) { return x.gt(&y) },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return `a` >= `b` lexicographically (Using partial order, `Ord`)
|
2014-02-24 08:11:00 -05:00
|
|
|
pub fn ge<A: Ord, T: Iterator<A>>(mut a: T, mut b: T) -> bool {
|
2013-08-08 22:07:21 +02:00
|
|
|
loop {
|
|
|
|
match (a.next(), b.next()) {
|
|
|
|
(None, None) => return true,
|
|
|
|
(None, _ ) => return false,
|
|
|
|
(_ , None) => return true,
|
|
|
|
(Some(x), Some(y)) => if x.ne(&y) { return x.ge(&y) },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-08-08 22:07:22 +02:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_lt() {
|
2014-03-08 18:11:52 -05:00
|
|
|
use slice::ImmutableVector;
|
2013-08-08 22:07:22 +02:00
|
|
|
|
|
|
|
let empty: [int, ..0] = [];
|
|
|
|
let xs = [1,2,3];
|
|
|
|
let ys = [1,2,0];
|
|
|
|
|
|
|
|
assert!(!lt(xs.iter(), ys.iter()));
|
|
|
|
assert!(!le(xs.iter(), ys.iter()));
|
|
|
|
assert!( gt(xs.iter(), ys.iter()));
|
|
|
|
assert!( ge(xs.iter(), ys.iter()));
|
|
|
|
|
|
|
|
assert!( lt(ys.iter(), xs.iter()));
|
|
|
|
assert!( le(ys.iter(), xs.iter()));
|
|
|
|
assert!(!gt(ys.iter(), xs.iter()));
|
|
|
|
assert!(!ge(ys.iter(), xs.iter()));
|
|
|
|
|
|
|
|
assert!( lt(empty.iter(), xs.iter()));
|
|
|
|
assert!( le(empty.iter(), xs.iter()));
|
|
|
|
assert!(!gt(empty.iter(), xs.iter()));
|
|
|
|
assert!(!ge(empty.iter(), xs.iter()));
|
|
|
|
|
|
|
|
// Sequence with NaN
|
|
|
|
let u = [1.0, 2.0];
|
|
|
|
let v = [0.0/0.0, 3.0];
|
|
|
|
|
|
|
|
assert!(!lt(u.iter(), v.iter()));
|
|
|
|
assert!(!le(u.iter(), v.iter()));
|
|
|
|
assert!(!gt(u.iter(), v.iter()));
|
|
|
|
assert!(!ge(u.iter(), v.iter()));
|
|
|
|
|
|
|
|
let a = [0.0/0.0];
|
|
|
|
let b = [1.0];
|
|
|
|
let c = [2.0];
|
|
|
|
|
|
|
|
assert!(lt(a.iter(), b.iter()) == (a[0] < b[0]));
|
|
|
|
assert!(le(a.iter(), b.iter()) == (a[0] <= b[0]));
|
|
|
|
assert!(gt(a.iter(), b.iter()) == (a[0] > b[0]));
|
|
|
|
assert!(ge(a.iter(), b.iter()) == (a[0] >= b[0]));
|
|
|
|
|
|
|
|
assert!(lt(c.iter(), b.iter()) == (c[0] < b[0]));
|
|
|
|
assert!(le(c.iter(), b.iter()) == (c[0] <= b[0]));
|
|
|
|
assert!(gt(c.iter(), b.iter()) == (c[0] > b[0]));
|
|
|
|
assert!(ge(c.iter(), b.iter()) == (c[0] >= b[0]));
|
|
|
|
}
|
2013-08-08 22:07:21 +02:00
|
|
|
}
|
|
|
|
|
2013-04-19 07:30:22 -04:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2014-05-01 18:06:59 -07:00
|
|
|
use realstd::prelude::*;
|
|
|
|
use realstd::iter::*;
|
|
|
|
use realstd::num;
|
2013-04-19 07:30:22 -04:00
|
|
|
|
2013-07-29 19:18:45 +02:00
|
|
|
use cmp;
|
2014-05-05 18:56:44 -07:00
|
|
|
use owned::Box;
|
2013-05-24 19:35:29 -07:00
|
|
|
use uint;
|
|
|
|
|
2013-04-24 19:54:13 -04:00
|
|
|
#[test]
|
2013-06-06 22:34:50 +02:00
|
|
|
fn test_counter_from_iter() {
|
2014-03-28 09:16:22 -07:00
|
|
|
let it = count(0, 5).take(10);
|
2014-03-30 21:45:55 -07:00
|
|
|
let xs: ~[int] = FromIterator::from_iter(it);
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(xs, box [0, 5, 10, 15, 20, 25, 30, 35, 40, 45]);
|
2013-04-24 19:54:13 -04:00
|
|
|
}
|
|
|
|
|
2013-04-19 11:29:38 -04:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_chain() {
|
|
|
|
let xs = [0u, 1, 2, 3, 4, 5];
|
2013-05-02 11:08:33 +09:00
|
|
|
let ys = [30u, 40, 50, 60];
|
2013-04-19 11:29:38 -04:00
|
|
|
let expected = [0, 1, 2, 3, 4, 5, 30, 40, 50, 60];
|
2013-08-09 20:22:59 -07:00
|
|
|
let mut it = xs.iter().chain(ys.iter());
|
2013-04-19 11:29:38 -04:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for &x in it {
|
2013-04-19 11:29:38 -04:00
|
|
|
assert_eq!(x, expected[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, expected.len());
|
2013-05-02 11:08:33 +09:00
|
|
|
|
2013-08-09 20:16:07 -07:00
|
|
|
let ys = count(30u, 10).take(4);
|
2013-08-09 20:22:59 -07:00
|
|
|
let mut it = xs.iter().map(|&x| x).chain(ys);
|
2013-05-02 11:08:33 +09:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in it {
|
2013-05-02 11:08:33 +09:00
|
|
|
assert_eq!(x, expected[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, expected.len());
|
2013-04-19 11:29:38 -04:00
|
|
|
}
|
|
|
|
|
2013-05-17 23:00:48 +09:00
|
|
|
#[test]
|
|
|
|
fn test_filter_map() {
|
2013-08-09 20:16:07 -07:00
|
|
|
let mut it = count(0u, 1u).take(10)
|
2014-02-17 07:20:01 +11:00
|
|
|
.filter_map(|x| if x % 2 == 0 { Some(x*x) } else { None });
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(it.collect::<~[uint]>(), box [0*0, 2*2, 4*4, 6*6, 8*8]);
|
2013-05-17 23:00:48 +09:00
|
|
|
}
|
|
|
|
|
2013-04-19 07:30:22 -04:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_enumerate() {
|
|
|
|
let xs = [0u, 1, 2, 3, 4, 5];
|
|
|
|
let mut it = xs.iter().enumerate();
|
2013-08-03 12:45:23 -04:00
|
|
|
for (i, &x) in it {
|
2013-04-19 07:30:22 -04:00
|
|
|
assert_eq!(i, x);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-08 20:46:36 +01:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_peekable() {
|
2014-04-25 01:08:02 -07:00
|
|
|
let xs = box [0u, 1, 2, 3, 4, 5];
|
2013-08-08 20:46:36 +01:00
|
|
|
let mut it = xs.iter().map(|&x|x).peekable();
|
|
|
|
assert_eq!(it.peek().unwrap(), &0);
|
|
|
|
assert_eq!(it.next().unwrap(), 0);
|
|
|
|
assert_eq!(it.next().unwrap(), 1);
|
|
|
|
assert_eq!(it.next().unwrap(), 2);
|
|
|
|
assert_eq!(it.peek().unwrap(), &3);
|
|
|
|
assert_eq!(it.peek().unwrap(), &3);
|
|
|
|
assert_eq!(it.next().unwrap(), 3);
|
|
|
|
assert_eq!(it.next().unwrap(), 4);
|
|
|
|
assert_eq!(it.peek().unwrap(), &5);
|
|
|
|
assert_eq!(it.next().unwrap(), 5);
|
|
|
|
assert!(it.peek().is_none());
|
|
|
|
assert!(it.next().is_none());
|
|
|
|
}
|
|
|
|
|
2013-04-19 07:30:22 -04:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_take_while() {
|
|
|
|
let xs = [0u, 1, 2, 3, 5, 13, 15, 16, 17, 19];
|
|
|
|
let ys = [0u, 1, 2, 3, 5, 13];
|
|
|
|
let mut it = xs.iter().take_while(|&x| *x < 15u);
|
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for &x in it {
|
2013-04-19 07:30:22 -04:00
|
|
|
assert_eq!(x, ys[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, ys.len());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_skip_while() {
|
|
|
|
let xs = [0u, 1, 2, 3, 5, 13, 15, 16, 17, 19];
|
|
|
|
let ys = [15, 16, 17, 19];
|
|
|
|
let mut it = xs.iter().skip_while(|&x| *x < 15u);
|
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for &x in it {
|
2013-04-19 07:30:22 -04:00
|
|
|
assert_eq!(x, ys[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, ys.len());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_skip() {
|
|
|
|
let xs = [0u, 1, 2, 3, 5, 13, 15, 16, 17, 19, 20, 30];
|
|
|
|
let ys = [13, 15, 16, 17, 19, 20, 30];
|
|
|
|
let mut it = xs.iter().skip(5);
|
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for &x in it {
|
2013-04-19 07:30:22 -04:00
|
|
|
assert_eq!(x, ys[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, ys.len());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_take() {
|
|
|
|
let xs = [0u, 1, 2, 3, 5, 13, 15, 16, 17, 19];
|
|
|
|
let ys = [0u, 1, 2, 3, 5];
|
2013-08-09 20:16:07 -07:00
|
|
|
let mut it = xs.iter().take(5);
|
2013-04-19 07:30:22 -04:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for &x in it {
|
2013-04-19 07:30:22 -04:00
|
|
|
assert_eq!(x, ys[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, ys.len());
|
|
|
|
}
|
2013-04-20 00:07:07 +10:00
|
|
|
|
2013-04-20 13:32:27 +10:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_scan() {
|
|
|
|
// test the type inference
|
2013-09-26 02:26:09 -04:00
|
|
|
fn add(old: &mut int, new: &uint) -> Option<f64> {
|
2013-04-20 13:32:27 +10:00
|
|
|
*old += *new as int;
|
2013-09-26 02:26:09 -04:00
|
|
|
Some(*old as f64)
|
2013-04-20 13:32:27 +10:00
|
|
|
}
|
|
|
|
let xs = [0u, 1, 2, 3, 4];
|
2013-09-26 02:26:09 -04:00
|
|
|
let ys = [0f64, 1.0, 3.0, 6.0, 10.0];
|
2013-04-20 13:32:27 +10:00
|
|
|
|
|
|
|
let mut it = xs.iter().scan(0, add);
|
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in it {
|
2013-04-20 13:32:27 +10:00
|
|
|
assert_eq!(x, ys[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, ys.len());
|
|
|
|
}
|
|
|
|
|
2013-06-24 02:34:38 +02:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_flat_map() {
|
|
|
|
let xs = [0u, 3, 6];
|
|
|
|
let ys = [0u, 1, 2, 3, 4, 5, 6, 7, 8];
|
2013-08-09 20:20:05 -07:00
|
|
|
let mut it = xs.iter().flat_map(|&x| count(x, 1).take(3));
|
2013-06-24 02:34:38 +02:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for x in it {
|
2013-06-24 02:34:38 +02:00
|
|
|
assert_eq!(x, ys[i]);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, ys.len());
|
|
|
|
}
|
|
|
|
|
2013-07-05 13:00:11 -04:00
|
|
|
#[test]
|
2013-08-11 01:55:34 +02:00
|
|
|
fn test_inspect() {
|
2013-07-05 13:00:11 -04:00
|
|
|
let xs = [1u, 2, 3, 4];
|
|
|
|
let mut n = 0;
|
|
|
|
|
|
|
|
let ys = xs.iter()
|
2013-08-09 20:09:47 -07:00
|
|
|
.map(|&x| x)
|
2013-08-11 01:55:34 +02:00
|
|
|
.inspect(|_| n += 1)
|
2013-07-05 13:00:11 -04:00
|
|
|
.collect::<~[uint]>();
|
|
|
|
|
|
|
|
assert_eq!(n, xs.len());
|
2014-02-28 01:23:06 -08:00
|
|
|
assert_eq!(xs.as_slice(), ys.as_slice());
|
2013-07-05 13:00:11 -04:00
|
|
|
}
|
|
|
|
|
2013-04-20 00:07:07 +10:00
|
|
|
#[test]
|
|
|
|
fn test_unfoldr() {
|
|
|
|
fn count(st: &mut uint) -> Option<uint> {
|
|
|
|
if *st < 10 {
|
|
|
|
let ret = Some(*st);
|
|
|
|
*st += 1;
|
|
|
|
ret
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-08 10:52:19 +10:00
|
|
|
let mut it = Unfold::new(0, count);
|
2013-04-20 00:07:07 +10:00
|
|
|
let mut i = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for counted in it {
|
2013-04-20 00:07:07 +10:00
|
|
|
assert_eq!(counted, i);
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(i, 10);
|
|
|
|
}
|
2013-05-17 23:45:25 +09:00
|
|
|
|
2013-07-18 17:34:28 +02:00
|
|
|
#[test]
|
|
|
|
fn test_cycle() {
|
|
|
|
let cycle_len = 3;
|
2013-08-09 20:16:07 -07:00
|
|
|
let it = count(0u, 1).take(cycle_len).cycle();
|
2014-01-25 20:37:51 +13:00
|
|
|
assert_eq!(it.size_hint(), (uint::MAX, None));
|
2013-08-09 20:16:07 -07:00
|
|
|
for (i, x) in it.take(100).enumerate() {
|
2013-07-18 17:34:28 +02:00
|
|
|
assert_eq!(i % cycle_len, x);
|
|
|
|
}
|
|
|
|
|
2013-08-09 20:16:07 -07:00
|
|
|
let mut it = count(0u, 1).take(0).cycle();
|
2013-07-18 17:34:28 +02:00
|
|
|
assert_eq!(it.size_hint(), (0, Some(0)));
|
|
|
|
assert_eq!(it.next(), None);
|
|
|
|
}
|
|
|
|
|
2013-05-17 23:45:25 +09:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_nth() {
|
|
|
|
let v = &[0, 1, 2, 3, 4];
|
2013-08-03 12:45:23 -04:00
|
|
|
for i in range(0u, v.len()) {
|
2013-05-18 04:27:58 -04:00
|
|
|
assert_eq!(v.iter().nth(i).unwrap(), &v[i]);
|
2013-05-17 23:45:25 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_last() {
|
|
|
|
let v = &[0, 1, 2, 3, 4];
|
2013-08-09 20:25:45 -07:00
|
|
|
assert_eq!(v.iter().last().unwrap(), &4);
|
|
|
|
assert_eq!(v.slice(0, 1).iter().last().unwrap(), &0);
|
2013-05-17 23:45:25 +09:00
|
|
|
}
|
2013-05-17 23:54:58 +09:00
|
|
|
|
|
|
|
#[test]
|
2013-06-17 19:43:22 -04:00
|
|
|
fn test_iterator_len() {
|
2013-05-17 23:54:58 +09:00
|
|
|
let v = &[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
2013-08-09 20:30:03 -07:00
|
|
|
assert_eq!(v.slice(0, 4).iter().len(), 4);
|
|
|
|
assert_eq!(v.slice(0, 10).iter().len(), 10);
|
|
|
|
assert_eq!(v.slice(0, 0).iter().len(), 0);
|
2013-05-17 23:54:58 +09:00
|
|
|
}
|
2013-05-18 00:18:09 +09:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_sum() {
|
|
|
|
let v = &[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
2013-08-09 20:09:47 -07:00
|
|
|
assert_eq!(v.slice(0, 4).iter().map(|&x| x).sum(), 6);
|
|
|
|
assert_eq!(v.iter().map(|&x| x).sum(), 55);
|
|
|
|
assert_eq!(v.slice(0, 0).iter().map(|&x| x).sum(), 0);
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_product() {
|
|
|
|
let v = &[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
2013-08-09 20:09:47 -07:00
|
|
|
assert_eq!(v.slice(0, 4).iter().map(|&x| x).product(), 0);
|
|
|
|
assert_eq!(v.slice(1, 5).iter().map(|&x| x).product(), 24);
|
|
|
|
assert_eq!(v.slice(0, 0).iter().map(|&x| x).product(), 1);
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_max() {
|
|
|
|
let v = &[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
2013-08-09 20:09:47 -07:00
|
|
|
assert_eq!(v.slice(0, 4).iter().map(|&x| x).max(), Some(3));
|
|
|
|
assert_eq!(v.iter().map(|&x| x).max(), Some(10));
|
|
|
|
assert_eq!(v.slice(0, 0).iter().map(|&x| x).max(), None);
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_iterator_min() {
|
|
|
|
let v = &[0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
|
2013-08-09 20:09:47 -07:00
|
|
|
assert_eq!(v.slice(0, 4).iter().map(|&x| x).min(), Some(0));
|
|
|
|
assert_eq!(v.iter().map(|&x| x).min(), Some(0));
|
|
|
|
assert_eq!(v.slice(0, 0).iter().map(|&x| x).min(), None);
|
2013-05-18 00:18:09 +09:00
|
|
|
}
|
|
|
|
|
2013-07-02 20:59:26 -07:00
|
|
|
#[test]
|
|
|
|
fn test_iterator_size_hint() {
|
2013-08-05 20:02:26 -04:00
|
|
|
let c = count(0, 1);
|
2013-07-02 20:59:26 -07:00
|
|
|
let v = &[0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
|
|
|
|
let v2 = &[10, 11, 12];
|
|
|
|
let vi = v.iter();
|
|
|
|
|
2014-01-25 20:37:51 +13:00
|
|
|
assert_eq!(c.size_hint(), (uint::MAX, None));
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(vi.size_hint(), (10, Some(10)));
|
|
|
|
|
2013-08-09 20:16:07 -07:00
|
|
|
assert_eq!(c.take(5).size_hint(), (5, Some(5)));
|
2014-02-16 20:36:43 +11:00
|
|
|
assert_eq!(c.skip(5).size_hint().val1(), None);
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(c.take_while(|_| false).size_hint(), (0, None));
|
|
|
|
assert_eq!(c.skip_while(|_| false).size_hint(), (0, None));
|
2014-01-25 20:37:51 +13:00
|
|
|
assert_eq!(c.enumerate().size_hint(), (uint::MAX, None));
|
|
|
|
assert_eq!(c.chain(vi.map(|&i| i)).size_hint(), (uint::MAX, None));
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(c.zip(vi).size_hint(), (10, Some(10)));
|
|
|
|
assert_eq!(c.scan(0, |_,_| Some(0)).size_hint(), (0, None));
|
|
|
|
assert_eq!(c.filter(|_| false).size_hint(), (0, None));
|
2014-01-25 20:37:51 +13:00
|
|
|
assert_eq!(c.map(|_| 0).size_hint(), (uint::MAX, None));
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(c.filter_map(|_| Some(0)).size_hint(), (0, None));
|
|
|
|
|
2013-08-09 20:16:07 -07:00
|
|
|
assert_eq!(vi.take(5).size_hint(), (5, Some(5)));
|
|
|
|
assert_eq!(vi.take(12).size_hint(), (10, Some(10)));
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(vi.skip(3).size_hint(), (7, Some(7)));
|
|
|
|
assert_eq!(vi.skip(12).size_hint(), (0, Some(0)));
|
|
|
|
assert_eq!(vi.take_while(|_| false).size_hint(), (0, Some(10)));
|
|
|
|
assert_eq!(vi.skip_while(|_| false).size_hint(), (0, Some(10)));
|
|
|
|
assert_eq!(vi.enumerate().size_hint(), (10, Some(10)));
|
2013-08-09 20:22:59 -07:00
|
|
|
assert_eq!(vi.chain(v2.iter()).size_hint(), (13, Some(13)));
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(vi.zip(v2.iter()).size_hint(), (3, Some(3)));
|
|
|
|
assert_eq!(vi.scan(0, |_,_| Some(0)).size_hint(), (0, Some(10)));
|
|
|
|
assert_eq!(vi.filter(|_| false).size_hint(), (0, Some(10)));
|
2013-08-09 20:09:47 -07:00
|
|
|
assert_eq!(vi.map(|i| i+1).size_hint(), (10, Some(10)));
|
2013-07-02 20:59:26 -07:00
|
|
|
assert_eq!(vi.filter_map(|_| Some(0)).size_hint(), (0, Some(10)));
|
|
|
|
}
|
|
|
|
|
2013-06-03 23:48:52 +02:00
|
|
|
#[test]
|
|
|
|
fn test_collect() {
|
2014-04-25 01:08:02 -07:00
|
|
|
let a = box [1, 2, 3, 4, 5];
|
2013-08-09 20:09:47 -07:00
|
|
|
let b: ~[int] = a.iter().map(|&x| x).collect();
|
2013-06-03 23:48:52 +02:00
|
|
|
assert_eq!(a, b);
|
|
|
|
}
|
|
|
|
|
2013-05-18 00:24:43 +09:00
|
|
|
#[test]
|
|
|
|
fn test_all() {
|
2014-05-05 18:56:44 -07:00
|
|
|
let v: Box<&[int]> = box &[1, 2, 3, 4, 5];
|
2013-06-08 14:07:55 +10:00
|
|
|
assert!(v.iter().all(|&x| x < 10));
|
2014-02-17 07:20:01 +11:00
|
|
|
assert!(!v.iter().all(|&x| x % 2 == 0));
|
2013-06-08 14:07:55 +10:00
|
|
|
assert!(!v.iter().all(|&x| x > 100));
|
2013-10-21 13:08:31 -07:00
|
|
|
assert!(v.slice(0, 0).iter().all(|_| fail!()));
|
2013-05-18 00:24:43 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_any() {
|
2014-05-05 18:56:44 -07:00
|
|
|
let v: Box<&[int]> = box &[1, 2, 3, 4, 5];
|
2013-07-04 22:13:26 -04:00
|
|
|
assert!(v.iter().any(|&x| x < 10));
|
2014-02-17 07:20:01 +11:00
|
|
|
assert!(v.iter().any(|&x| x % 2 == 0));
|
2013-07-04 22:13:26 -04:00
|
|
|
assert!(!v.iter().any(|&x| x > 100));
|
2013-10-21 13:08:31 -07:00
|
|
|
assert!(!v.slice(0, 0).iter().any(|_| fail!()));
|
2013-05-18 00:24:43 +09:00
|
|
|
}
|
2013-06-15 17:42:31 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_find() {
|
2013-08-09 07:19:23 -07:00
|
|
|
let v: &[int] = &[1, 3, 9, 27, 103, 14, 11];
|
2013-08-09 20:49:29 -07:00
|
|
|
assert_eq!(*v.iter().find(|x| *x & 1 == 0).unwrap(), 14);
|
|
|
|
assert_eq!(*v.iter().find(|x| *x % 3 == 0).unwrap(), 3);
|
|
|
|
assert!(v.iter().find(|x| *x % 12 == 0).is_none());
|
2013-06-15 17:42:31 -04:00
|
|
|
}
|
2013-06-15 17:56:26 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_position() {
|
|
|
|
let v = &[1, 3, 9, 27, 103, 14, 11];
|
2013-07-04 22:13:26 -04:00
|
|
|
assert_eq!(v.iter().position(|x| *x & 1 == 0).unwrap(), 5);
|
|
|
|
assert_eq!(v.iter().position(|x| *x % 3 == 0).unwrap(), 1);
|
|
|
|
assert!(v.iter().position(|x| *x % 12 == 0).is_none());
|
2013-06-15 17:56:26 -04:00
|
|
|
}
|
2013-06-17 19:43:22 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_count() {
|
|
|
|
let xs = &[1, 2, 2, 1, 5, 9, 0, 2];
|
|
|
|
assert_eq!(xs.iter().count(|x| *x == 2), 3);
|
|
|
|
assert_eq!(xs.iter().count(|x| *x == 5), 1);
|
|
|
|
assert_eq!(xs.iter().count(|x| *x == 95), 0);
|
|
|
|
}
|
2013-06-27 06:51:04 +09:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_max_by() {
|
2013-08-09 07:19:23 -07:00
|
|
|
let xs: &[int] = &[-3, 0, 1, 5, -10];
|
2013-06-27 06:51:04 +09:00
|
|
|
assert_eq!(*xs.iter().max_by(|x| x.abs()).unwrap(), -10);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_min_by() {
|
2013-08-09 07:19:23 -07:00
|
|
|
let xs: &[int] = &[-3, 0, 1, 5, -10];
|
2013-06-27 06:51:04 +09:00
|
|
|
assert_eq!(*xs.iter().min_by(|x| x.abs()).unwrap(), 0);
|
|
|
|
}
|
2013-07-11 00:23:59 -04:00
|
|
|
|
2013-09-30 20:59:46 +02:00
|
|
|
#[test]
|
|
|
|
fn test_by_ref() {
|
|
|
|
let mut xs = range(0, 10);
|
|
|
|
// sum the first five values
|
|
|
|
let partial_sum = xs.by_ref().take(5).fold(0, |a, b| a + b);
|
|
|
|
assert_eq!(partial_sum, 10);
|
|
|
|
assert_eq!(xs.next(), Some(5));
|
|
|
|
}
|
|
|
|
|
2013-07-11 00:23:59 -04:00
|
|
|
#[test]
|
2014-01-23 20:41:57 +01:00
|
|
|
fn test_rev() {
|
2013-07-11 00:23:59 -04:00
|
|
|
let xs = [2, 4, 6, 8, 10, 12, 14, 16];
|
|
|
|
let mut it = xs.iter();
|
|
|
|
it.next();
|
|
|
|
it.next();
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(it.rev().map(|&x| x).collect::<~[int]>(), box [16, 14, 12, 10, 8, 6]);
|
2013-07-11 00:23:59 -04:00
|
|
|
}
|
2013-07-18 19:50:51 -04:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_double_ended_map() {
|
|
|
|
let xs = [1, 2, 3, 4, 5, 6];
|
2013-08-09 20:09:47 -07:00
|
|
|
let mut it = xs.iter().map(|&x| x * -1);
|
2013-07-18 19:50:51 -04:00
|
|
|
assert_eq!(it.next(), Some(-1));
|
|
|
|
assert_eq!(it.next(), Some(-2));
|
|
|
|
assert_eq!(it.next_back(), Some(-6));
|
|
|
|
assert_eq!(it.next_back(), Some(-5));
|
|
|
|
assert_eq!(it.next(), Some(-3));
|
|
|
|
assert_eq!(it.next_back(), Some(-4));
|
|
|
|
assert_eq!(it.next(), None);
|
|
|
|
}
|
|
|
|
|
2013-08-30 20:00:11 +02:00
|
|
|
#[test]
|
|
|
|
fn test_double_ended_enumerate() {
|
|
|
|
let xs = [1, 2, 3, 4, 5, 6];
|
|
|
|
let mut it = xs.iter().map(|&x| x).enumerate();
|
|
|
|
assert_eq!(it.next(), Some((0, 1)));
|
|
|
|
assert_eq!(it.next(), Some((1, 2)));
|
|
|
|
assert_eq!(it.next_back(), Some((5, 6)));
|
|
|
|
assert_eq!(it.next_back(), Some((4, 5)));
|
|
|
|
assert_eq!(it.next_back(), Some((3, 4)));
|
|
|
|
assert_eq!(it.next_back(), Some((2, 3)));
|
|
|
|
assert_eq!(it.next(), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_double_ended_zip() {
|
|
|
|
let xs = [1, 2, 3, 4, 5, 6];
|
|
|
|
let ys = [1, 2, 3, 7];
|
|
|
|
let a = xs.iter().map(|&x| x);
|
|
|
|
let b = ys.iter().map(|&x| x);
|
|
|
|
let mut it = a.zip(b);
|
|
|
|
assert_eq!(it.next(), Some((1, 1)));
|
|
|
|
assert_eq!(it.next(), Some((2, 2)));
|
|
|
|
assert_eq!(it.next_back(), Some((4, 7)));
|
|
|
|
assert_eq!(it.next_back(), Some((3, 3)));
|
|
|
|
assert_eq!(it.next(), None);
|
|
|
|
}
|
|
|
|
|
2013-07-18 19:50:51 -04:00
|
|
|
#[test]
|
|
|
|
fn test_double_ended_filter() {
|
|
|
|
let xs = [1, 2, 3, 4, 5, 6];
|
|
|
|
let mut it = xs.iter().filter(|&x| *x & 1 == 0);
|
|
|
|
assert_eq!(it.next_back().unwrap(), &6);
|
|
|
|
assert_eq!(it.next_back().unwrap(), &4);
|
|
|
|
assert_eq!(it.next().unwrap(), &2);
|
|
|
|
assert_eq!(it.next_back(), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_double_ended_filter_map() {
|
|
|
|
let xs = [1, 2, 3, 4, 5, 6];
|
|
|
|
let mut it = xs.iter().filter_map(|&x| if x & 1 == 0 { Some(x * 2) } else { None });
|
|
|
|
assert_eq!(it.next_back().unwrap(), 12);
|
|
|
|
assert_eq!(it.next_back().unwrap(), 8);
|
|
|
|
assert_eq!(it.next().unwrap(), 4);
|
|
|
|
assert_eq!(it.next_back(), None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_double_ended_chain() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
2014-04-25 01:08:02 -07:00
|
|
|
let ys = box [7, 9, 11];
|
2014-01-23 20:41:57 +01:00
|
|
|
let mut it = xs.iter().chain(ys.iter()).rev();
|
2013-07-18 19:50:51 -04:00
|
|
|
assert_eq!(it.next().unwrap(), &11)
|
|
|
|
assert_eq!(it.next().unwrap(), &9)
|
|
|
|
assert_eq!(it.next_back().unwrap(), &1)
|
|
|
|
assert_eq!(it.next_back().unwrap(), &2)
|
|
|
|
assert_eq!(it.next_back().unwrap(), &3)
|
|
|
|
assert_eq!(it.next_back().unwrap(), &4)
|
|
|
|
assert_eq!(it.next_back().unwrap(), &5)
|
|
|
|
assert_eq!(it.next_back().unwrap(), &7)
|
|
|
|
assert_eq!(it.next_back(), None)
|
|
|
|
}
|
2013-07-22 20:11:24 -04:00
|
|
|
|
2013-09-01 16:43:47 +02:00
|
|
|
#[test]
|
|
|
|
fn test_rposition() {
|
|
|
|
fn f(xy: &(int, char)) -> bool { let (_x, y) = *xy; y == 'b' }
|
|
|
|
fn g(xy: &(int, char)) -> bool { let (_x, y) = *xy; y == 'd' }
|
2014-04-25 01:08:02 -07:00
|
|
|
let v = box [(0, 'a'), (1, 'b'), (2, 'c'), (3, 'b')];
|
2013-09-01 16:43:47 +02:00
|
|
|
|
|
|
|
assert_eq!(v.iter().rposition(f), Some(3u));
|
|
|
|
assert!(v.iter().rposition(g).is_none());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
|
|
|
fn test_rposition_fail() {
|
2014-04-25 01:08:02 -07:00
|
|
|
let v = [(box 0, @0), (box 0, @0), (box 0, @0), (box 0, @0)];
|
2013-09-01 16:43:47 +02:00
|
|
|
let mut i = 0;
|
2013-11-21 17:23:21 -08:00
|
|
|
v.iter().rposition(|_elt| {
|
2013-09-01 16:43:47 +02:00
|
|
|
if i == 2 {
|
2013-10-21 13:08:31 -07:00
|
|
|
fail!()
|
2013-09-01 16:43:47 +02:00
|
|
|
}
|
|
|
|
i += 1;
|
|
|
|
false
|
2013-11-21 17:23:21 -08:00
|
|
|
});
|
2013-09-01 16:43:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-29 19:18:45 +02:00
|
|
|
#[cfg(test)]
|
|
|
|
fn check_randacc_iter<A: Eq, T: Clone + RandomAccessIterator<A>>(a: T, len: uint)
|
|
|
|
{
|
|
|
|
let mut b = a.clone();
|
|
|
|
assert_eq!(len, b.indexable());
|
|
|
|
let mut n = 0;
|
2013-08-03 12:45:23 -04:00
|
|
|
for (i, elt) in a.enumerate() {
|
2014-02-28 01:23:06 -08:00
|
|
|
assert!(Some(elt) == b.idx(i));
|
2013-07-29 19:18:45 +02:00
|
|
|
n += 1;
|
|
|
|
}
|
|
|
|
assert_eq!(n, len);
|
2014-02-28 01:23:06 -08:00
|
|
|
assert!(None == b.idx(n));
|
2013-07-29 19:18:45 +02:00
|
|
|
// call recursively to check after picking off an element
|
|
|
|
if len > 0 {
|
|
|
|
b.next();
|
|
|
|
check_randacc_iter(b, len-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-20 03:11:28 +02:00
|
|
|
#[test]
|
|
|
|
fn test_double_ended_flat_map() {
|
|
|
|
let u = [0u,1];
|
|
|
|
let v = [5,6,7,8];
|
2013-08-09 20:20:05 -07:00
|
|
|
let mut it = u.iter().flat_map(|x| v.slice(*x, v.len()).iter());
|
2013-07-20 03:11:28 +02:00
|
|
|
assert_eq!(it.next_back().unwrap(), &8);
|
|
|
|
assert_eq!(it.next().unwrap(), &5);
|
|
|
|
assert_eq!(it.next_back().unwrap(), &7);
|
|
|
|
assert_eq!(it.next_back().unwrap(), &6);
|
|
|
|
assert_eq!(it.next_back().unwrap(), &8);
|
|
|
|
assert_eq!(it.next().unwrap(), &6);
|
|
|
|
assert_eq!(it.next_back().unwrap(), &7);
|
|
|
|
assert_eq!(it.next_back(), None);
|
|
|
|
assert_eq!(it.next(), None);
|
|
|
|
assert_eq!(it.next_back(), None);
|
|
|
|
}
|
|
|
|
|
2013-07-22 20:11:24 -04:00
|
|
|
#[test]
|
|
|
|
fn test_random_access_chain() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
2014-04-25 01:08:02 -07:00
|
|
|
let ys = box [7, 9, 11];
|
2013-08-09 20:22:59 -07:00
|
|
|
let mut it = xs.iter().chain(ys.iter());
|
2013-07-22 20:11:24 -04:00
|
|
|
assert_eq!(it.idx(0).unwrap(), &1);
|
|
|
|
assert_eq!(it.idx(5).unwrap(), &7);
|
|
|
|
assert_eq!(it.idx(7).unwrap(), &11);
|
|
|
|
assert!(it.idx(8).is_none());
|
|
|
|
|
|
|
|
it.next();
|
|
|
|
it.next();
|
|
|
|
it.next_back();
|
|
|
|
|
|
|
|
assert_eq!(it.idx(0).unwrap(), &3);
|
|
|
|
assert_eq!(it.idx(4).unwrap(), &9);
|
|
|
|
assert!(it.idx(6).is_none());
|
2013-07-29 19:18:45 +02:00
|
|
|
|
|
|
|
check_randacc_iter(it, xs.len() + ys.len() - 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_random_access_enumerate() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
check_randacc_iter(xs.iter().enumerate(), xs.len());
|
|
|
|
}
|
|
|
|
|
2013-08-03 19:40:20 +02:00
|
|
|
#[test]
|
2014-01-23 20:41:57 +01:00
|
|
|
fn test_random_access_rev() {
|
2013-08-03 19:40:20 +02:00
|
|
|
let xs = [1, 2, 3, 4, 5];
|
2014-01-23 20:41:57 +01:00
|
|
|
check_randacc_iter(xs.iter().rev(), xs.len());
|
|
|
|
let mut it = xs.iter().rev();
|
2013-08-03 19:40:20 +02:00
|
|
|
it.next();
|
|
|
|
it.next_back();
|
|
|
|
it.next();
|
|
|
|
check_randacc_iter(it, xs.len() - 3);
|
|
|
|
}
|
|
|
|
|
2013-07-29 19:18:45 +02:00
|
|
|
#[test]
|
|
|
|
fn test_random_access_zip() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
let ys = [7, 9, 11];
|
|
|
|
check_randacc_iter(xs.iter().zip(ys.iter()), cmp::min(xs.len(), ys.len()));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_random_access_take() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
let empty: &[int] = [];
|
2013-08-09 20:16:07 -07:00
|
|
|
check_randacc_iter(xs.iter().take(3), 3);
|
|
|
|
check_randacc_iter(xs.iter().take(20), xs.len());
|
|
|
|
check_randacc_iter(xs.iter().take(0), 0);
|
|
|
|
check_randacc_iter(empty.iter().take(2), 0);
|
2013-07-29 19:18:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_random_access_skip() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
let empty: &[int] = [];
|
|
|
|
check_randacc_iter(xs.iter().skip(2), xs.len() - 2);
|
|
|
|
check_randacc_iter(empty.iter().skip(2), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-08-11 01:55:34 +02:00
|
|
|
fn test_random_access_inspect() {
|
2013-07-29 19:18:45 +02:00
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
|
2013-08-11 01:55:34 +02:00
|
|
|
// test .map and .inspect that don't implement Clone
|
2014-04-21 23:25:18 -07:00
|
|
|
let mut it = xs.iter().inspect(|_| {});
|
2013-07-29 19:18:45 +02:00
|
|
|
assert_eq!(xs.len(), it.indexable());
|
2013-08-03 12:45:23 -04:00
|
|
|
for (i, elt) in xs.iter().enumerate() {
|
2013-07-29 19:18:45 +02:00
|
|
|
assert_eq!(Some(elt), it.idx(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-08-09 20:09:47 -07:00
|
|
|
fn test_random_access_map() {
|
2013-07-29 19:18:45 +02:00
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
|
2014-04-21 23:25:18 -07:00
|
|
|
let mut it = xs.iter().map(|x| *x);
|
2013-07-29 19:18:45 +02:00
|
|
|
assert_eq!(xs.len(), it.indexable());
|
2013-08-03 12:45:23 -04:00
|
|
|
for (i, elt) in xs.iter().enumerate() {
|
2013-07-29 19:18:45 +02:00
|
|
|
assert_eq!(Some(*elt), it.idx(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_random_access_cycle() {
|
|
|
|
let xs = [1, 2, 3, 4, 5];
|
|
|
|
let empty: &[int] = [];
|
2013-08-09 20:16:07 -07:00
|
|
|
check_randacc_iter(xs.iter().cycle().take(27), 27);
|
2013-07-29 19:18:45 +02:00
|
|
|
check_randacc_iter(empty.iter().cycle(), 0);
|
2013-07-22 20:11:24 -04:00
|
|
|
}
|
2013-08-06 22:34:22 -07:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_double_ended_range() {
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(range(11i, 14).rev().collect::<~[int]>(), box [13i, 12, 11]);
|
2014-01-23 20:41:57 +01:00
|
|
|
for _ in range(10i, 0).rev() {
|
2013-10-21 13:08:31 -07:00
|
|
|
fail!("unreachable");
|
2013-08-06 22:34:22 -07:00
|
|
|
}
|
|
|
|
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(range(11u, 14).rev().collect::<~[uint]>(), box [13u, 12, 11]);
|
2014-01-23 20:41:57 +01:00
|
|
|
for _ in range(10u, 0).rev() {
|
2013-10-21 13:08:31 -07:00
|
|
|
fail!("unreachable");
|
2013-08-06 22:34:22 -07:00
|
|
|
}
|
|
|
|
}
|
2013-08-17 18:41:53 -04:00
|
|
|
|
2013-09-15 00:54:32 -04:00
|
|
|
#[test]
|
|
|
|
fn test_range() {
|
2013-11-11 05:06:26 -05:00
|
|
|
/// A mock type to check Range when ToPrimitive returns None
|
|
|
|
struct Foo;
|
|
|
|
|
|
|
|
impl ToPrimitive for Foo {
|
|
|
|
fn to_i64(&self) -> Option<i64> { None }
|
|
|
|
fn to_u64(&self) -> Option<u64> { None }
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Add<Foo, Foo> for Foo {
|
|
|
|
fn add(&self, _: &Foo) -> Foo {
|
|
|
|
Foo
|
|
|
|
}
|
2014-02-24 08:11:00 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for Foo {
|
|
|
|
fn eq(&self, _: &Foo) -> bool {
|
|
|
|
true
|
|
|
|
}
|
2013-11-11 05:06:26 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Ord for Foo {
|
|
|
|
fn lt(&self, _: &Foo) -> bool {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Clone for Foo {
|
|
|
|
fn clone(&self) -> Foo {
|
|
|
|
Foo
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-18 17:08:23 +11:00
|
|
|
impl Mul<Foo, Foo> for Foo {
|
|
|
|
fn mul(&self, _: &Foo) -> Foo {
|
|
|
|
Foo
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-11 05:06:26 -05:00
|
|
|
impl num::One for Foo {
|
|
|
|
fn one() -> Foo {
|
|
|
|
Foo
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(range(0i, 5).collect::<~[int]>(), box [0i, 1, 2, 3, 4]);
|
|
|
|
assert_eq!(range(-10i, -1).collect::<~[int]>(), box [-10, -9, -8, -7, -6, -5, -4, -3, -2]);
|
|
|
|
assert_eq!(range(0i, 5).rev().collect::<~[int]>(), box [4, 3, 2, 1, 0]);
|
|
|
|
assert_eq!(range(200, -5).collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range(200, -5).rev().collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range(200, 200).collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range(200, 200).rev().collect::<~[int]>(), box []);
|
2013-11-11 05:06:26 -05:00
|
|
|
|
|
|
|
assert_eq!(range(0i, 100).size_hint(), (100, Some(100)));
|
|
|
|
// this test is only meaningful when sizeof uint < sizeof u64
|
2014-01-25 20:37:51 +13:00
|
|
|
assert_eq!(range(uint::MAX - 1, uint::MAX).size_hint(), (1, Some(1)));
|
2013-11-11 05:06:26 -05:00
|
|
|
assert_eq!(range(-10i, -1).size_hint(), (9, Some(9)));
|
|
|
|
assert_eq!(range(Foo, Foo).size_hint(), (0, None));
|
2013-09-15 00:54:32 -04:00
|
|
|
}
|
|
|
|
|
2013-08-17 18:41:53 -04:00
|
|
|
#[test]
|
|
|
|
fn test_range_inclusive() {
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(range_inclusive(0i, 5).collect::<~[int]>(), box [0i, 1, 2, 3, 4, 5]);
|
|
|
|
assert_eq!(range_inclusive(0i, 5).rev().collect::<~[int]>(), box [5i, 4, 3, 2, 1, 0]);
|
|
|
|
assert_eq!(range_inclusive(200, -5).collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range_inclusive(200, -5).rev().collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range_inclusive(200, 200).collect::<~[int]>(), box [200]);
|
|
|
|
assert_eq!(range_inclusive(200, 200).rev().collect::<~[int]>(), box [200]);
|
2013-08-17 18:41:53 -04:00
|
|
|
}
|
2013-08-19 17:52:20 -04:00
|
|
|
|
2013-09-14 16:47:21 -04:00
|
|
|
#[test]
|
|
|
|
fn test_range_step() {
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(range_step(0i, 20, 5).collect::<~[int]>(), box [0, 5, 10, 15]);
|
|
|
|
assert_eq!(range_step(20i, 0, -5).collect::<~[int]>(), box [20, 15, 10, 5]);
|
|
|
|
assert_eq!(range_step(20i, 0, -6).collect::<~[int]>(), box [20, 14, 8, 2]);
|
|
|
|
assert_eq!(range_step(200u8, 255, 50).collect::<~[u8]>(), box [200u8, 250]);
|
|
|
|
assert_eq!(range_step(200, -5, 1).collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range_step(200, 200, 1).collect::<~[int]>(), box []);
|
2013-09-14 16:47:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_range_step_inclusive() {
|
2014-04-25 01:08:02 -07:00
|
|
|
assert_eq!(range_step_inclusive(0i, 20, 5).collect::<~[int]>(), box [0, 5, 10, 15, 20]);
|
|
|
|
assert_eq!(range_step_inclusive(20i, 0, -5).collect::<~[int]>(), box [20, 15, 10, 5, 0]);
|
|
|
|
assert_eq!(range_step_inclusive(20i, 0, -6).collect::<~[int]>(), box [20, 14, 8, 2]);
|
|
|
|
assert_eq!(range_step_inclusive(200u8, 255, 50).collect::<~[u8]>(), box [200u8, 250]);
|
|
|
|
assert_eq!(range_step_inclusive(200, -5, 1).collect::<~[int]>(), box []);
|
|
|
|
assert_eq!(range_step_inclusive(200, 200, 1).collect::<~[int]>(), box [200]);
|
2013-09-14 16:47:21 -04:00
|
|
|
}
|
|
|
|
|
2013-08-19 17:52:20 -04:00
|
|
|
#[test]
|
|
|
|
fn test_reverse() {
|
|
|
|
let mut ys = [1, 2, 3, 4, 5];
|
|
|
|
ys.mut_iter().reverse_();
|
2014-02-28 01:23:06 -08:00
|
|
|
assert!(ys == [5, 4, 3, 2, 1]);
|
2013-08-19 17:52:20 -04:00
|
|
|
}
|
2014-01-11 14:13:06 +04:00
|
|
|
|
2014-01-25 02:44:06 -05:00
|
|
|
#[test]
|
2014-01-14 18:38:06 +04:00
|
|
|
fn test_peekable_is_empty() {
|
2014-01-11 14:13:06 +04:00
|
|
|
let a = [1];
|
|
|
|
let mut it = a.iter().peekable();
|
2014-01-14 18:38:06 +04:00
|
|
|
assert!( !it.is_empty() );
|
2014-01-11 14:13:06 +04:00
|
|
|
it.next();
|
2014-01-14 18:38:06 +04:00
|
|
|
assert!( it.is_empty() );
|
2014-01-11 14:13:06 +04:00
|
|
|
}
|
2014-01-24 22:40:54 -08:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_min_max() {
|
|
|
|
let v: [int, ..0] = [];
|
|
|
|
assert_eq!(v.iter().min_max(), NoElements);
|
|
|
|
|
|
|
|
let v = [1i];
|
|
|
|
assert!(v.iter().min_max() == OneElement(&1));
|
|
|
|
|
|
|
|
let v = [1i, 2, 3, 4, 5];
|
|
|
|
assert!(v.iter().min_max() == MinMax(&1, &5));
|
|
|
|
|
|
|
|
let v = [1i, 2, 3, 4, 5, 6];
|
|
|
|
assert!(v.iter().min_max() == MinMax(&1, &6));
|
|
|
|
|
|
|
|
let v = [1i, 1, 1, 1];
|
|
|
|
assert!(v.iter().min_max() == MinMax(&1, &1));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_MinMaxResult() {
|
|
|
|
let r: MinMaxResult<int> = NoElements;
|
|
|
|
assert_eq!(r.into_option(), None)
|
|
|
|
|
|
|
|
let r = OneElement(1);
|
|
|
|
assert_eq!(r.into_option(), Some((1,1)));
|
|
|
|
|
|
|
|
let r = MinMax(1,2);
|
|
|
|
assert_eq!(r.into_option(), Some((1,2)));
|
|
|
|
}
|
2013-04-19 07:30:22 -04:00
|
|
|
}
|