使用迭代器处理一系列条目

迭代器模式允许你依次对一系列条目执行某个任务。迭代器负责迭代每个条目的逻辑并确定序列何时完成。当你使用迭代器时,你不必自己重新实现该逻辑。

在 Rust 中,迭代器是惰性的,这意味着在你调用消耗迭代器以使用它的方法之前,它们不会产生任何效果。例如,清单 13-10 中的代码通过调用 Vec<T> 上定义的 iter 方法,创建一个遍历向量 v1 中条目的迭代器。这段代码本身并没有做任何有用的事情。

文件名:src/main.rs
fn main() {
    let v1 = vec![1, 2, 3];

    let v1_iter = v1.iter();
}
清单 13-10:创建一个迭代器

迭代器存储在 v1_iter 变量中。一旦我们创建了一个迭代器,我们就可以通过多种方式使用它。在第 3 章的清单 3-5 中,我们使用 for 循环遍历一个数组,以便对其每个条目执行一些代码。在底层,这隐式地创建然后消耗了一个迭代器,但我们一直忽略了它具体是如何工作的,直到现在。

在清单 13-11 中的示例中,我们将迭代器的创建与 for 循环中迭代器的使用分开。当使用 v1_iter 中的迭代器调用 for 循环时,迭代器中的每个元素都会在循环的一次迭代中使用,从而打印出每个值。

文件名:src/main.rs
fn main() {
    let v1 = vec![1, 2, 3];

    let v1_iter = v1.iter();

    for val in v1_iter {
        println!("Got: {val}");
    }
}
清单 13-11:在 for 循环中使用迭代器

在标准库没有提供迭代器的语言中,你可能会通过将一个变量从索引 0 开始,使用该变量索引到向量以获取值,并在循环中递增变量值直到达到向量中的条目总数来编写相同的功能。

迭代器为你处理所有这些逻辑,从而减少了你可能弄错的重复代码。迭代器为你提供了更大的灵活性,可以将相同的逻辑用于多种不同的序列,而不仅仅是你可以索引的数据结构,例如向量。让我们研究一下迭代器是如何做到这一点的。

Iterator 特征和 next 方法

所有迭代器都实现了一个名为 Iterator 的特征,该特征在标准库中定义。该特征的定义如下:

#![allow(unused)]
fn main() {
pub trait Iterator {
    type Item;

    fn next(&mut self) -> Option<Self::Item>;

    // methods with default implementations elided
}
}

请注意,此定义使用了一些新的语法:type ItemSelf::Item,它们正在定义此特征的关联类型。我们将在第 19 章中深入讨论关联类型。现在,你只需要知道此代码表示实现 Iterator 特征还需要你定义一个 Item 类型,并且此 Item 类型用于 next 方法的返回类型中。换句话说,Item 类型将是从迭代器返回的类型。

Iterator 特征只要求实现者定义一个方法:next 方法,它一次返回迭代器的一个条目,并用 Some 包裹,当迭代结束时,返回 None

我们可以直接在迭代器上调用 next 方法;清单 13-12 演示了从对向量创建的迭代器重复调用 next 返回的值。

文件名:src/lib.rs
#[cfg(test)]
mod tests {
    #[test]
    fn iterator_demonstration() {
        let v1 = vec![1, 2, 3];

        let mut v1_iter = v1.iter();

        assert_eq!(v1_iter.next(), Some(&1));
        assert_eq!(v1_iter.next(), Some(&2));
        assert_eq!(v1_iter.next(), Some(&3));
        assert_eq!(v1_iter.next(), None);
    }
}
清单 13-12:在迭代器上调用 next 方法

请注意,我们需要使 v1_iter 可变:在迭代器上调用 next 方法会更改迭代器用于跟踪它在序列中的位置的内部状态。换句话说,此代码消耗或耗尽了迭代器。每次调用 next 都会从迭代器中吃掉一个条目。当我们使用 for 循环时,我们不需要使 v1_iter 可变,因为循环获取了 v1_iter 的所有权并在幕后使其可变。

另请注意,我们从调用 next 中获得的值是向量中值的不可变引用。iter 方法会生成一个遍历不可变引用的迭代器。如果我们想要创建一个获取 v1 所有权并返回拥有值的迭代器,我们可以调用 into_iter 而不是 iter。类似地,如果我们想要遍历可变引用,我们可以调用 iter_mut 而不是 iter

消耗迭代器的方法

Iterator 特征有许多不同的方法,标准库提供了默认实现;你可以通过查看标准库 API 文档中的 Iterator 特征来了解这些方法。其中一些方法在其定义中调用 next 方法,这就是为什么你需要在实现 Iterator 特征时实现 next 方法的原因。

调用 next 的方法称为消耗适配器,因为调用它们会耗尽迭代器。一个示例是 sum 方法,它获取迭代器的所有权,并通过重复调用 next 来遍历条目,从而消耗迭代器。在遍历过程中,它将每个条目添加到运行总计中,并在迭代完成时返回总计。清单 13-13 有一个测试,说明了 sum 方法的使用

文件名:src/lib.rs
#[cfg(test)]
mod tests {
    #[test]
    fn iterator_sum() {
        let v1 = vec![1, 2, 3];

        let v1_iter = v1.iter();

        let total: i32 = v1_iter.sum();

        assert_eq!(total, 6);
    }
}
清单 13-13:调用 sum 方法以获取迭代器中所有条目的总和

在调用 sum 之后,我们不允许使用 v1_iter,因为 sum 获取了我们调用它的迭代器的所有权。

生成其他迭代器的方法

迭代器适配器是在 Iterator 特征上定义的方法,它们不消耗迭代器。相反,它们通过更改原始迭代器的某些方面来生成不同的迭代器。

清单 13-14 显示了调用迭代器适配器方法 map 的示例,该方法接受一个闭包,以便在遍历条目时调用每个条目。map 方法返回一个新的迭代器,该迭代器生成修改后的条目。此处的闭包创建一个新的迭代器,其中向量中的每个条目将递增 1

文件名:src/main.rs
fn main() {
    let v1: Vec<i32> = vec![1, 2, 3];

    v1.iter().map(|x| x + 1);
}
清单 13-14:调用迭代器适配器 map 以创建一个新的迭代器

但是,此代码会产生警告

$ cargo run
   Compiling iterators v0.1.0 (file:///projects/iterators)
warning: unused `Map` that must be used
 --> src/main.rs:4:5
  |
4 |     v1.iter().map(|x| x + 1);
  |     ^^^^^^^^^^^^^^^^^^^^^^^^
  |
  = note: iterators are lazy and do nothing unless consumed
  = note: `#[warn(unused_must_use)]` on by default
help: use `let _ = ...` to ignore the resulting value
  |
4 |     let _ = v1.iter().map(|x| x + 1);
  |     +++++++

warning: `iterators` (bin "iterators") generated 1 warning
    Finished `dev` profile [unoptimized + debuginfo] target(s) in 0.47s
     Running `target/debug/iterators`

清单 13-14 中的代码没有任何作用;我们指定的闭包永远不会被调用。该警告提醒我们原因:迭代器适配器是惰性的,我们需要在此处消耗迭代器。

要解决此警告并消耗迭代器,我们将使用 collect 方法,我们在第 12 章中将 env::args 与清单 12-1 一起使用。此方法消耗迭代器并将结果值收集到集合数据类型中。

在清单 13-15 中,我们将遍历从调用 map 返回的迭代器的结果收集到一个向量中。此向量最终将包含原始向量中的每个条目,并递增 1。

文件名:src/main.rs
fn main() {
    let v1: Vec<i32> = vec![1, 2, 3];

    let v2: Vec<_> = v1.iter().map(|x| x + 1).collect();

    assert_eq!(v2, vec![2, 3, 4]);
}
清单 13-15:调用 map 方法以创建一个新的迭代器,然后调用 collect 方法以消耗新的迭代器并创建一个向量

由于 map 采用闭包,我们可以指定想要对每个条目执行的任何操作。这是一个很好的示例,说明闭包如何让你自定义某些行为,同时重用 Iterator 特征提供的迭代行为。

你可以链接对迭代器适配器的多次调用,以可读的方式执行复杂的操作。但是,由于所有迭代器都是惰性的,因此你必须调用其中一个消耗适配器方法才能从对迭代器适配器的调用中获得结果。

使用捕获其环境的闭包

许多迭代器适配器将闭包作为参数,并且通常我们将指定为迭代器适配器参数的闭包是捕获其环境的闭包。

对于此示例,我们将使用采用闭包的 filter 方法。闭包从迭代器中获取一个条目并返回一个 bool。如果闭包返回 true,则该值将包含在 filter 生成的迭代中。如果闭包返回 false,则该值将不包括在内。

在清单 13-16 中,我们将 filter 与捕获其环境中 shoe_size 变量的闭包一起使用,以遍历 Shoe 结构实例的集合。它将仅返回指定大小的鞋子。

文件名:src/lib.rs
#[derive(PartialEq, Debug)]
struct Shoe {
    size: u32,
    style: String,
}

fn shoes_in_size(shoes: Vec<Shoe>, shoe_size: u32) -> Vec<Shoe> {
    shoes.into_iter().filter(|s| s.size == shoe_size).collect()
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn filters_by_size() {
        let shoes = vec![
            Shoe {
                size: 10,
                style: String::from("sneaker"),
            },
            Shoe {
                size: 13,
                style: String::from("sandal"),
            },
            Shoe {
                size: 10,
                style: String::from("boot"),
            },
        ];

        let in_my_size = shoes_in_size(shoes, 10);

        assert_eq!(
            in_my_size,
            vec![
                Shoe {
                    size: 10,
                    style: String::from("sneaker")
                },
                Shoe {
                    size: 10,
                    style: String::from("boot")
                },
            ]
        );
    }
}
清单 13-16:将 filter 方法与捕获 shoe_size 的闭包一起使用

shoes_in_size 函数获取鞋子向量和鞋子尺寸作为参数的所有权。它返回一个仅包含指定尺寸鞋子的向量。

shoes_in_size 的主体中,我们调用 into_iter 来创建一个获取向量所有权的迭代器。然后,我们调用 filter 以将该迭代器调整为一个新的迭代器,该迭代器仅包含闭包返回 true 的元素。

闭包从环境中捕获 shoe_size 参数,并将该值与每只鞋子的尺寸进行比较,仅保留指定尺寸的鞋子。最后,调用 collect 将调整后的迭代器返回的值收集到一个向量中,该向量由函数返回。

该测试表明,当我们调用 shoes_in_size 时,我们只会获得与我们指定的值尺寸相同的鞋子。