如何 运行 for 循环向量的元素,并在 for 循环内部和 for 循环外部更改向量?

How to run for loop on elements of a vector and change the vector inside the for loop and outside the for loop in rust?

我是 Rust 的新手。我需要在 for 循环之前创建一个向量。 运行for循环就可以了。更改 for 循环内的向量。然后在 for 循环之后更改向量。

我尝试了以下代码并尝试使用不可变借用,但都没有用。

fn main() {
    let mut vec1 = vec![4, 5];
    vec1.push(6);
    for i in vec1 {
        if i % 2 == 0 {
            vec1.push(7);
        }
    }
    vec1.push(8);
    println!("vec1={:?}", vec1);
}

我希望在 for 循环内部和之后编译和更改向量。但它显示此错误消息:

error[E0382]: borrow of moved value: `vec1`
 --> src/main.rs:6:13
  |
2 |     let mut vec1 = vec![4, 5];
  |         -------- move occurs because `vec1` has type `std::vec::Vec<i32>`, which does not implement the `Copy` trait
3 |     vec1.push(6);
4 |     for i in vec1 {
  |              ----
  |              |
  |              value moved here
  |              help: consider borrowing to avoid moving into the for loop: `&vec1`
5 |         if i % 2 == 0 {
6 |             vec1.push(7);
  |             ^^^^ value borrowed here after move

你能解释一下为什么会发生移动吗?你能编译它吗?

您的代码有两个问题。幸运的是,这两个问题都可以通过改变一件事来解决。问题是:

  • 写入 for _ in vec1 将值 vec1 移动到循环中。这是一个与其他所有动作一样的动作,这意味着在循环后无法访问该值!如果您需要复习所有权和搬迁,请阅读 the related chapter in the Rust book。您可以通过 for _ in &vec1.
  • 迭代对向量元素的引用
  • 您正试图改变您正在迭代的向量。无论您是按值(正如您所做的那样)还是按引用迭代向量,都不能在迭代向量时更改向量。这有很多充分的理由!在您的情况下,如果您在迭代时添加元素,很容易导致无限循环。

要同时解决这两个问题,您可以迭代向量的索引而不是向量元素 (Playground):

let mut vec1 = vec![4, 5];
vec1.push(6);
for i in 0..vec1.len() {
    if vec1[i] % 2 == 0 {
        vec1.push(7);
    }
}
vec1.push(8);
println!("vec1={:?}", vec1);

这样,向量不会被 for 循环移动或借用,我们可以在循环期间和之后自由地改变它。这个特定的解决方案迭代原始向量的索引,这意味着循环中添加的元素不会被循环迭代。这是避免意外无限循环的良好保护。但是请注意,您仍然可以搬起石头砸自己的脚,例如,在迭代期间从向量中删除元素。一般来说,独立于编程语言,在迭代集合时改变集合是危险的,应该小心完成。

你可以像这样用循环来做,但是要小心。(Playground)

fn main() {
    let mut vec1 = vec![4, 5];
    vec1.push(6);
    
    let mut i = 0;
    loop {
        if i >= vec1.len() { break; }
        if vec1[i] % 2 == 0 {
            vec1.push(7);
        }
        
        i += 1;
    }
    vec1.push(8);
    println!("vec1={:?}", vec1);
}