Rust 目前提供 3 种方法来进行一些迭代操作。他们是loop
,while
和for
。每种方法都有自己的用途。
无限loop
是 Rust 提供的最简单的循环。使用loop
关键字,Rust 提供了一个直到一些终止语句被执行的循环方法。Rust 的无限loop
看起来像这样:
loop {
println!("Loop forever!");
}
Rust 也有一个while
循环。它看起来像:
let mut x = 5; // mut x: i32
let mut done = false; // mut done: bool
while !done {
x += x - 3;
println!("{}", x);
if x % 5 == 0 {
done = true;
}
}
while
循环是当你不确定应该循环多少次时正确的选择。
如果你需要一个无限循环,你可能想要这么写:
while true {
然而,loop
远比它适合处理这个情况:
loop {
Rust 的控制流分析会区别对待这个与while true
,因为我们知道它会一直循环。现阶段理解这些细节意味着什么并不是非常重要,基本上,你给编译器越多的信息,越能确保安全和生成更好的代码,所以当你打算无限循环的时候应该总是倾向于使用loop
。
for
用来循环一个特定的次数。然而,Rust的for
循环与其它系统语言有些许不同。Rust的for
循环看起来并不像这个“C语言样式”的for
循环:
for (x = 0; x < 10; x++) {
printf( "%d\n", x );
}
相反,它看起来像这个样子:
for x in 0..10 {
println!("{}", x); // x: i32
}
更抽象的形式:
for var in expression {
code
}
这个表达式是一个[迭代器](Iterators 迭代器.md).迭代器返回一系列的元素。每个元素是循环中的一次重复。然后它的值与var
绑定,它在循环体中有效。每当循环体执行完后,我们从迭代器中取出下一个值,然后我们再重复一遍。当迭代器中不再有值时,for
循环结束。
在我们的例子中,0..10
表达式取一个开始和结束的位置,然后给出一个含有这之间值得迭代器。当然它不包括上限值,所以我们的循环会打印0
到9
,而不是到10
。
Rust 没有使用“C语言风格”的for
循环是有意为之的。即使对于有经验的 C 语言开发者来说,要手动控制要循环的每个元素也都是复杂并且易于出错的。
当你需要记录你已经循环了多少次了的时候,你可以使用.enumerate()
函数。
for (i,j) in (5..10).enumerate() {
println!("i = {} and j = {}", i, j);
}
输出:
i = 0 and j = 5
i = 1 and j = 6
i = 2 and j = 7
i = 3 and j = 8
i = 4 and j = 9
别忘了在范围外面加上括号。
# let lines = "hello\nworld".lines();
for (linenumber, line) in lines.enumerate() {
println!("{}: {}", linenumber, line);
}
输出:
0: hello
1: world
让我们再看一眼之前的while
循环:
let mut x = 5;
let mut done = false;
while !done {
x += x - 3;
println!("{}", x);
if x % 5 == 0 {
done = true;
}
}
我们必须使用一个mut
布尔型变量绑定,done
,来确定何时我们应该推出循环。Rust 有两个关键字帮助我们来修改迭代:break
和continue
。
这样,我们可以用break
来写一个更好的循环:
let mut x = 5;
loop {
x += x - 3;
println!("{}", x);
if x % 5 == 0 { break; }
}
现在我们用loop
来无限循环,然后用break
来提前退出循环。
continue
比较类似,不过不是退出循环,它直接进行下一次迭代。下面的例子只会打印奇数:
for x in 0..10 {
if x % 2 == 0 { continue; }
println!("{}", x);
}
break
和continue
在while
循环和for
循环中都有效。
你也许会遇到这样的情形,当你有嵌套的循环而希望指定你的哪一个break
或continue
该起作用。就像大多数语言,默认break
或continue
将会作用于最内层的循环。当你想要一个break
或continue
作用于一个外层循环,你可以使用标签来指定你的break
或continue
语句作用的循环。如下代码只会在x
和y
都为奇数时打印他们:
'outer: for x in 0..10 {
'inner: for y in 0..10 {
if x % 2 == 0 { continue 'outer; } // continues the loop over x
if y % 2 == 0 { continue 'inner; } // continues the loop over y
println!("x: {}, y: {}", x, y);
}
}