【问题标题】:borrowed value does not live long enough in loop借来的价值在循环中的寿命不够长
【发布时间】:2019-11-24 20:45:42
【问题描述】:

我正在尝试解析文件并从函数返回Vec<Vec<&str>>。但是在推送到向量时,我在文件读取循环中遇到了借用值错误。

use std::io::{self, BufReader, prelude::*};
use std::fs::File;

fn read() -> Vec<Vec<&'static str>> {
 let file = File::open("~/test").expect("failed to read file");
 let reader = BufReader::new(file);
 let mut main_vector: Vec<Vec<&str>> = Vec::new();
    for line in reader.lines() {
        match line {
            Ok(v) => {
                let mut sub_vector: Vec<&str> = Vec::new();
                for element in v.split_whitespace().collect::<Vec<&str>>() {
                    sub_vector.push(element);
                }
                main_vector.push(sub_vector);
            },
            Err(e) => panic!("failed to parse: {:?}", e),
        }
    }
    //return main_vector;
}

这是编译器错误:

error[E0597]: `v` does not live long enough
  --> src/main.rs:67:32
   |
67 |                 for element in v.split_whitespace().collect::<Vec<&str>>() {
   |                                ^ borrowed value does not live long enough
...
70 |                 main_vector.push(sub_vector);
   |                 -------------- borrow later used here
71 |             },
   |              - `v` dropped here while still borrowed

我认为这是关于参考和借用,但我仍然很难弄清楚这一点。

【问题讨论】:

    标签: reference rust lifetime borrowing


    【解决方案1】:

    如果提供更多代码会更好,但我尝试复制它并到达这个工作的sn-p:

    type Error = i32;
    
    struct Reader
    {
        lines: Vec<Result<String, Error>>
    }
    
    impl Reader
    {
        pub fn new() -> Self
        {
            Self{lines: vec![Ok("foo".into()), Ok("bar".into())]}
        }
    
        pub fn lines(&self) -> &[Result<String, Error>]
        {
            &self.lines
        }
    }
    
    fn main() {
        let reader = Reader::new();
        let mut main_vector: Vec<Vec<&str>> = Vec::new();
        for line in reader.lines() {
            match line {
                Ok(v) => {
                    let mut sub_vector: Vec<&str> = Vec::new();
                    for element in v.split_whitespace().collect::<Vec<&str>>() {
                        sub_vector.push(element);
                    }
                    main_vector.push(sub_vector);
                },
                Err(e) => panic!("failed to parse: {:?}", e),
            }
        }
    }
    

    你可以在这里https://play.rust-lang.org/?version=stable&mode=debug&edition=2018&gist=f2785fcad682b9dd1f5ed61c7e0308d8Rust Playground 上查看它

    【讨论】:

    • 我已经添加了代码的其他部分。这个 sn-p 运行良好,但我认为 BufReader (.lines()) 是问题所在。
    【解决方案2】:

    这个问题类似于Return local String as a slice (&str)。最简单的解决方案与该问题中的相同 - 使用字符串,而不是 &str。 这些问题不同,因为该答案专门讨论从函数返回,而您没有列出函数。

    要解决生命周期导致代码失败的原因,请尝试一个更简单的示例

    fn main() {
        let mut v:Vec<&str> = Vec::new();
        {
            let chars = [b'x', b'y', b'z'];
            let s:&str = std::str::from_utf8(&chars).unwrap();
            v.push(&s);
         }
        println!("{:?}", v);
    }
    

    和编译器输出

    
    let s:&str = std::str::from_utf8(&chars).unwrap();
                                     ^^^^^^ borrowed value does not live long enough
    

    这不起作用的原因正是编译器所说的。 chars 在块内创建,因此它具有与该块关联的生命周期,并且当您的程序退出该块时,字符可能不再存在。任何引用chars 的东西都可能有一个悬空指针。 Rust 通过将其设为非法来避免悬空指针。在我的示例中,Rust 不允许这样做似乎很愚蠢,但在您的示例中,这是有道理的 - Rust 可以通过每次循环从 v.split_whitespace().collect::&lt;Vec&lt;&amp;str&gt;&gt;() 中删除旧的 strs 来保持堆栈较小。

    【讨论】:

    • 使用Vec&lt;Vec&lt;String&gt;&gt; 代替Vec&lt;Vec&lt;&amp;str&gt;&gt; 解决了这个问题。谢谢!
    • 感谢@orhun 让我们知道String 工作(而&amp;str 没有)。这也解决了我的问题。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2017-03-09
    • 2020-07-23
    • 1970-01-01
    相关资源
    最近更新 更多