由于生命周期/借用错误,文本文件解析功能无法编译

时间:2015-07-25 00:21:10

标签: rust text-parsing lifetime borrow-checker

NB。这篇文章原本是一个较大的帖子的一部分,其中包含两个问题(我认为是一个错误表现出来的不同),但为了遵守网站指南,我将其分成两个单独的帖子,其中第二个。第一篇文章是here

我正在尝试解析一个简单的配置文本文件,每行包含一个三字输入,其布局如下:

ITEM name value
ITEM name value
//etc.

我在这里(和on the Rust playground)重现了解析(以及后续编译错误)的函数:

use std::fs::File;
use std::io::prelude::*;
use std::io::BufReader;
use std::path::Path;
use std::collections::HashMap;

fn main() { }

pub fn parse(path: &Path) -> config_struct {

    let file = File::open(&path).unwrap();
    let reader = BufReader::new(&file);
    let line_iterator = reader.lines();
    let mut connection_map = HashMap::new();
    let mut target_map = HashMap::new();

    for line in line_iterator {

        let line_slice = line.unwrap();
        let word_vector: Vec<&str> = line_slice.split_whitespace().collect();

        if word_vector.len() != 3 { continue; }

        // no match statement   
        connection_map.insert(word_vector[1], word_vector[2]);
    }

    config_struct { connections: connection_map, targets: target_map }
}

pub struct config_struct<'a>  {
    // <name, value>
    connections: HashMap<&'a str, &'a str>,
    // <name, value>
    targets: HashMap<&'a str, &'a str>,
}
<anon>:20:38: 20:48 error: `line_slice` does not live long enough
<anon>:20         let word_vector: Vec<&str> = line_slice.split_whitespace().collect();
                                               ^~~~~~~~~~
note: in expansion of for loop expansion
<anon>:17:5: 26:6 note: expansion site
<anon>:9:44: 29:2 note: reference must be valid for the anonymous lifetime #1 defined on the block at 9:43...
<anon>:9 pub fn parse(path: &Path) -> config_struct {
<anon>:10 
<anon>:11     let file = File::open(&path).unwrap();
<anon>:12     let reader = BufReader::new(&file);
<anon>:13     let line_iterator = reader.lines();
<anon>:14     let mut connection_map = HashMap::new();
          ...
<anon>:19:40: 26:6 note: ...but borrowed value is only valid for the block suffix following statement 0 at 19:39
<anon>:19         let line_slice = line.unwrap();
<anon>:20         let word_vector: Vec<&str> = line_slice.split_whitespace().collect();
<anon>:21 
<anon>:22         if word_vector.len() != 3 { continue; }
<anon>:23 
<anon>:24         // no match statement   
          ...
error: aborting due to previous error

从本质上讲,我在借用检查器方面遇到了麻烦;在我的代码中,不是word_vector填充了不指向line_slice的拥有对象?我想可能unwrap()collect()返回了一个引用,并且它是超出范围的引用,但unwrapcollect的Rust文档建议不然。

1 个答案:

答案 0 :(得分:3)

如果没有存储它包含的值的东西,&str就不存在 - 它纯粹是一个引用(因此&)。

从文件中读取String个; 这些提供存储空间。但是你正在放弃它们,试图只返回字符串。

也可以这样想:

pub fn parse(path: &Path) -> config_struct<'???>;

您的退货价值应该是多少年?

它没有抱怨该部分的唯一原因是它推断出Path引用生命周期和返回值生命周期是相同的,这意味着你要返回对{{1}内部内容的引用{1}},你不是。

在这种情况下,您通常需要存储Path而不是String s。使用&str将每个&str转换为String