rust - 在bufreader.lines()的for循环后无法使用移动的BufReader

标签 rust traits

我试图从文件中读取一些行,跳过前几行并打印其余的行,但是在移动后,我不断收到有关已用值的错误:

use std::fs::File;
use std::io::{self, BufRead, BufReader, Read};
use std::path::Path;

fn skip_and_print_file(skip: &usize, path: &Path) {
    let mut skip: usize = *skip;

    if let Ok(file) = File::open(path) {
        let mut buffer = BufReader::new(file);
        for (index, line) in buffer.lines().enumerate() {
            if index >= skip {
                break;
            }
        }
        print_to_stdout(&mut buffer);
    }
}

fn print_to_stdout(mut input: &mut Read) {
    let mut stdout = io::stdout();
    io::copy(&mut input, &mut stdout);
}

fn main() {}

这是我得到的错误:

error[E0382]: use of moved value: `buffer`
  --> src/main.rs:15:30
   |
10 |         for (index, line) in buffer.lines().enumerate() {
   |                              ------ value moved here
...
15 |         print_to_stdout(&mut buffer);
   |                              ^^^^^^ value used here after move
   |
   = note: move occurs because `buffer` has type `std::io::BufReader<std::fs::File>`, which does not implement the `Copy` trait

最佳答案

作为Lukas Kalbertodt says,使用 Read::by_ref

这样可以防止lines使用BufReader,而是使用&mut BufReader。相同的逻辑applies to iterators

您可以使用 skip 来代替自己实现Iterator::take。但是,必须使用for循环将其驱动完成:

use std::{
    fs::File,
    io::{self, BufRead, BufReader, Read},
    path::Path,
};

fn skip_and_print_file(skip: usize, path: impl AsRef<Path>) {
    if let Ok(file) = File::open(path) {
        let mut buffer = BufReader::new(file);

        for _ in buffer.by_ref().lines().take(skip) {}
        // Or: buffer.by_ref().lines().take(skip).for_each(drop);

        print_to_stdout(buffer);
    }
}

fn print_to_stdout(mut input: impl Read) {
    let mut stdout = io::stdout();
    io::copy(&mut input, &mut stdout).expect("Unable to copy");
}

fn main() {
    skip_and_print_file(2, "/etc/hosts");
}

请注意,没有理由使skip变量可变甚至传递引用。您还可以输入AsRef<Path>,然后skip_and_print_file的调用者可以只传递字符串文字。

关于rust - 在bufreader.lines()的for循环后无法使用移动的BufReader,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64399394/

相关文章:

rust - 如何将迭代器适配器与返回impl Trait作为IntoIterator的IntoIter关联类型的函数一起使用?

rust - 动态生成参数Clap

rust - 如何在不消耗值的情况下实现添加特征

rust - 如何在 Futures 0.2 中合并 Streams?

c++ - 功能访问限制

reference - 如何从 &Vec<T> 或 Vec<&T> 创建 &T 的迭代器?

android - 如何从构建中排除特定的 crate 类型?

rust - 为什么 Rust 中的 Box 这么称呼?

C++检查模板参数的嵌套typedef以获得其标量基类型

rust - 如何在每个结构上自动#derive(Something)(可能带有过滤)?