借用的值在循环中的寿命不够长
borrowed value does not live long enough in loop
我正在尝试从函数中解析文件和 return 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
我认为这是关于参考和借用的,但我仍然很难弄清楚。
如果提供更多代码会更好,但我尝试重现它并得出了这个工作片段:
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),
}
}
}
你可以在 Rust Playground 上查看 https://play.rust-lang.org/?version=stable&mode=debug&edition=2018&gist=f2785fcad682b9dd1f5ed61c7e0308d8
此题与相似。最简单的解决方案与该问题相同 - 使用字符串,而不是 &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::<Vec<&str>>()
中删除旧的 str
s 来保持堆栈较小。
我正在尝试从函数中解析文件和 return 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
我认为这是关于参考和借用的,但我仍然很难弄清楚。
如果提供更多代码会更好,但我尝试重现它并得出了这个工作片段:
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),
}
}
}
你可以在 Rust Playground 上查看 https://play.rust-lang.org/?version=stable&mode=debug&edition=2018&gist=f2785fcad682b9dd1f5ed61c7e0308d8
此题与
要解决生命周期导致代码失败的原因,请尝试一个更简单的示例
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::<Vec<&str>>()
中删除旧的 str
s 来保持堆栈较小。