繁体   English   中英

如何在过滤器闭包中使用Peekable迭代器?

[英]How do I use a Peekable iterator in a filter closure?

我只需要找到下一个数字相同的数字: [1,2,2,3,4,4]应该产生[2,4] 因为我需要查看下一个数字,我想我会尝试使用Peekable迭代器并编写一个filter

fn main() {
    let xs = [1, 2, 2, 3, 4, 4];
    let mut iter = xs.iter().peekable();

    let pairs = iter.filter(move |num| {
        match iter.peek() {
            Some(next) => num == next,
            None       => false,
        }
    });

    for num in pairs {
        println!("{}", num);
    }
}

我收到一个错误:

error[E0382]: capture of moved value: `iter`
 --> src/main.rs:6:15
  |
5 |     let pairs = iter.filter(move |num| {
  |                 ---- value moved here
6 |         match iter.peek() {
  |               ^^^^ value captured here after move
  |
  = note: move occurs because `iter` has type `std::iter::Peekable<std::slice::Iter<'_, i32>>`, which does not implement the `Copy` trait

我认为这是因为闭包使用了iter ,但它没有借用它,也无法复制它。

如何解决这个想要在过滤器中引用迭代器的问题?

引用过滤器内的迭代器

我不相信你能。 当您调用filter ,它将获得基本迭代器的所有权:

fn filter<P>(self, predicate: P) -> Filter<Self, P> 
where
    P: FnMut(&Self::Item) -> bool, 

一旦你这样做,它就消失了。 有没有更多的iter 在某些类似的情况下,您可以使用Iterator::by_ref来可变地借用迭代器,将其驱动一段时间,然后返回原始版本。 在这种情况下,这将不起作用,因为内部迭代器需要第二次可变地借用它,这是不允许的。

找到下一个数字相同的数字。

extern crate itertools;

use itertools::Itertools;

fn main() {
    let input = [1, 2, 2, 3, 4, 4];

    let pairs = input
        .iter()
        .tuple_windows()
        .filter_map(|(a, b)| if a == b { Some(a) } else { None });

    let result: Vec<_> = pairs.cloned().collect();
    assert_eq!(result, [2, 4]);
}

或者如果你想要只使用标准库的东西:

fn main() {
    let xs = [1, 2, 2, 3, 4, 4];

    let mut prev = None;
    let pairs = xs.iter().filter_map(move |curr| {
        let next = if prev == Some(curr) { Some(curr) } else { None };
        prev = Some(curr);
        next
    });

    let result: Vec<_> = pairs.cloned().collect();
    assert_eq!(result, [2, 4]);
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM