for 循环
for 和 range
for in
结构可用于遍历 Iterator
。创建迭代器最简单的方法之一是使用区间表示法 a..b
。这会生成从 a
(包含)到 b
(不包含)的值,步长为 1。
让我们用 for
而不是 while
来编写 FizzBuzz。
fn main() { // `n` 在每次迭代中将取值:1, 2, ..., 100 for n in 1..101 { if n % 15 == 0 { println!("fizzbuzz"); } else if n % 3 == 0 { println!("fizz"); } else if n % 5 == 0 { println!("buzz"); } else { println!("{}", n); } } }
另外,可以使用 a..=b
表示两端都包含的范围。上面的代码可以改写为:
fn main() { // `n` 在每次迭代中将取值:1, 2, ..., 100 for n in 1..=100 { if n % 15 == 0 { println!("fizzbuzz"); } else if n % 3 == 0 { println!("fizz"); } else if n % 5 == 0 { println!("buzz"); } else { println!("{}", n); } } }
for 与迭代器
for in
结构能以多种方式与 Iterator
交互。正如在 Iterator 特质一节中讨论的那样,默认情况下 for
循环会对集合应用 into_iter
函数。然而,这并不是将集合转换为迭代器的唯一方法。
into_iter
、iter
和 iter_mut
都以不同的方式处理集合到迭代器的转换,通过提供对数据的不同视图。
iter
- 在每次迭代中借用集合的每个元素。因此,集合保持不变,并且在循环之后可以重复使用。
fn main() { let names = vec!["Bob", "Frank", "Ferris"]; for name in names.iter() { match name { &"Ferris" => println!("我们中间有一个 Rustacean!"), // TODO ^ 尝试删除 & 并只匹配 "Ferris" _ => println!("你好 {}", name), } } println!("names: {:?}", names); }
into_iter
- 这会消耗集合,使得在每次迭代中提供确切的数据。一旦集合被消耗,它就不再可用于重复使用,因为它已经在循环中被"移动"了。
fn main() { let names = vec!["Bob", "Frank", "Ferris"]; for name in names.into_iter() { match name { "Ferris" => println!("我们中间有一个 Rustacean!"), _ => println!("你好 {}", name), } } println!("names: {:?}", names); // 修复:^ 注释掉此行 }
iter_mut
- 这会可变地借用集合的每个元素,允许在原地修改集合。
fn main() { let mut names = vec!["Bob", "Frank", "Ferris"]; for name in names.iter_mut() { *name = match name { &mut "Ferris" => "我们中间有一个 Rustacean!", _ => "你好", } } println!("names: {:?}", names); }
在上面的代码片段中,注意 match
分支的类型,这是迭代类型的关键区别。类型的差异意味着可以执行不同的操作。