Skip to content

Commit a16626f

Browse files
committed
iter: Implement .fold() for .chain()
Chain can do something interesting here where it passes on the fold into its inner iterators. The lets the underlying iterator's custom fold() be used, and skips the regular chain logic in next.
1 parent 780acda commit a16626f

File tree

2 files changed

+31
-0
lines changed

2 files changed

+31
-0
lines changed

src/libcore/iter/mod.rs

+19
Original file line numberDiff line numberDiff line change
@@ -550,6 +550,25 @@ impl<A, B> Iterator for Chain<A, B> where
550550
}
551551
}
552552

553+
fn fold<Acc, F>(self, init: Acc, mut f: F) -> Acc
554+
where F: FnMut(Acc, Self::Item) -> Acc,
555+
{
556+
let mut accum = init;
557+
match self.state {
558+
ChainState::Both | ChainState::Front => {
559+
accum = self.a.fold(accum, &mut f);
560+
}
561+
_ => { }
562+
}
563+
match self.state {
564+
ChainState::Both | ChainState::Back => {
565+
accum = self.b.fold(accum, &mut f);
566+
}
567+
_ => { }
568+
}
569+
accum
570+
}
571+
553572
#[inline]
554573
fn nth(&mut self, mut n: usize) -> Option<A::Item> {
555574
match self.state {

src/libcoretest/iter.rs

+12
Original file line numberDiff line numberDiff line change
@@ -985,6 +985,18 @@ fn test_empty() {
985985
assert_eq!(it.next(), None);
986986
}
987987

988+
#[test]
989+
fn test_chain_fold() {
990+
let xs = [1, 2, 3];
991+
let ys = [1, 2, 0];
992+
993+
let mut iter = xs.iter().chain(&ys);
994+
iter.next();
995+
let mut result = Vec::new();
996+
iter.fold((), |(), &elt| result.push(elt));
997+
assert_eq!(&[2, 3, 1, 2, 0], &result[..]);
998+
}
999+
9881000
#[bench]
9891001
fn bench_rposition(b: &mut Bencher) {
9901002
let it: Vec<usize> = (0..300).collect();

0 commit comments

Comments
 (0)