From a0c2949e7c8c67fc7739482de8cee374a253b523 Mon Sep 17 00:00:00 2001 From: Huon Wilson Date: Sat, 20 Apr 2013 13:32:27 +1000 Subject: [PATCH] libcore: add a ScanIterator, a generalisation of MapIterator to have internal state. --- src/libcore/iterator.rs | 45 +++++++++++++++++++++++++++++++++++++++-- 1 file changed, 43 insertions(+), 2 deletions(-) diff --git a/src/libcore/iterator.rs b/src/libcore/iterator.rs index 4929b1b8dba37..4a3a98df0d0ea 100644 --- a/src/libcore/iterator.rs +++ b/src/libcore/iterator.rs @@ -39,6 +39,8 @@ pub trait IteratorUtil { fn take_while<'r>(self, predicate: &'r fn(&A) -> bool) -> TakeWhileIterator<'r, A, Self>; fn skip(self, n: uint) -> SkipIterator; fn take(self, n: uint) -> TakeIterator; + fn scan<'r, St, B>(self, initial_state: St, f: &'r fn(&mut St, A) -> Option) + -> ScanIterator<'r, A, B, Self, St>; fn advance(&mut self, f: &fn(A) -> bool); } @@ -93,6 +95,12 @@ impl> IteratorUtil for T { TakeIterator{iter: self, n: n} } + #[inline(always)] + fn scan<'r, St, B>(self, initial_state: St, f: &'r fn(&mut St, A) -> Option) + -> ScanIterator<'r, A, B, T, St> { + ScanIterator{iter: self, f: f, state: initial_state} + } + /// A shim implementing the `for` loop iteration protocol for iterator objects #[inline] fn advance(&mut self, f: &fn(A) -> bool) { @@ -306,12 +314,13 @@ impl> Iterator for TakeIterator { pub struct UnfoldrIterator<'self, A, St> { priv f: &'self fn(&mut St) -> Option, - priv state: St + state: St } pub impl<'self, A, St> UnfoldrIterator<'self, A, St> { #[inline] - fn new(f: &'self fn(&mut St) -> Option, initial_state: St) -> UnfoldrIterator<'self, A, St> { + fn new(f: &'self fn(&mut St) -> Option, initial_state: St) + -> UnfoldrIterator<'self, A, St> { UnfoldrIterator { f: f, state: initial_state @@ -326,6 +335,19 @@ impl<'self, A, St> Iterator for UnfoldrIterator<'self, A, St> { } } +pub struct ScanIterator<'self, A, B, T, St> { + priv iter: T, + priv f: &'self fn(&mut St, A) -> Option, + state: St +} + +impl<'self, A, B, T: Iterator, St> Iterator for ScanIterator<'self, A, B, T, St> { + #[inline] + fn next(&mut self) -> Option { + self.iter.next().chain(|a| (self.f)(&mut self.state, a)) + } +} + #[cfg(test)] mod tests { use super::*; @@ -406,6 +428,25 @@ mod tests { assert_eq!(i, ys.len()); } + #[test] + fn test_iterator_scan() { + // test the type inference + fn add(old: &mut int, new: &uint) -> Option { + *old += *new as int; + Some(*old as float) + } + let xs = [0u, 1, 2, 3, 4]; + let ys = [0f, 1f, 3f, 6f, 10f]; + + let mut it = xs.iter().scan(0, add); + let mut i = 0; + for it.advance |x| { + assert_eq!(x, ys[i]); + i += 1; + } + assert_eq!(i, ys.len()); + } + #[test] fn test_unfoldr() { fn count(st: &mut uint) -> Option {