最近我一直在尝试开发《 super 星际迷航》游戏的 Rust 版本。但是,我在加载保存的游戏时遇到了一个小问题。
要加载游戏,我将保存文件读入字符串,将其拆分为 Vec<&str> at (以\0x1e 作为分隔符),然后在两个结果元素上使用 serde_json::from_str 来获取我的游戏数据对象(一个用于企业,另一个用于宇宙)。
不幸的是,当我尝试退回它们时,出现以下错误:
57 | return Some((ent, uni))
| ^^^^^^^^^^^^^^^^ returns a value referencing data owned by the current function
据我所知,这是因为 .split.collect() 中的数据称为“临时数据”。
如何才能返回结构?
编辑:这是 github 存储库:https://github.com/thescribe11/Rust-SST
我的代码:
use crate::structs::{Enterprise, Universe};
use std::fs::{File};
use std::io::{Read, Write, stdin, stdout};
use serde_json::{to_string, from_str};
<snip>
pub fn get_config_from_file<'a> () -> Option<(Enterprise, Universe<'a>)>{
//! Thaw a game.
//!
//! The .sst file type is as follows:
//! <json data for Enterprise object>\0x1e<json data for Universe object>
let mut save_file: File;
loop {
let temp = File::open(input("Save file: "));
match temp {
Ok(p) => {save_file = p; break;},
Err(e) => {println!("Unable to find save file.\n"); continue;}
};
}
let pass = input("Password: ");
let mut enc_data = String::new();
match save_file.read_to_string(&mut enc_data) {
Ok(_) => {},
Err(_) => {println!("\nERROR: The save file is corrupted."); return None}
}
let mut ent: Enterprise; let mut uni: Universe;
let raw_parts: Vec<&str> = enc_data.split("\0x1e").collect();
ent = match serde_json::from_str(raw_parts[0]) {
Ok(data) => {data},
Err(_) => {println!("\nERROR: The save file is corrupted."); return None}
};
uni = match serde_json::from_str(raw_parts[1]) {
Ok(data) => data,
Err(_) => {println!("\nERROR: The save file is corrupted."); return None}
};
return Some((ent, uni))
}
最佳答案
我将通过删除与问题无关的所有内容来极大地简化您的代码:
use serde::{Deserialize};
#[derive(Debug, Deserialize)]
pub struct Universe<'a> {
#[serde(borrow)]
player_name: Vec<&'a str>,
}
pub fn get_config_from_file<'a> () -> Option<Universe<'a>> {
let raw: String = "some_data".to_string();
let uni: Universe = serde_json::from_str(&raw).unwrap();
return Some(uni);
}
这仍然给出以下错误消息:
error[E0515]: cannot return value referencing local variable `raw`
--> src/lib.rs:12:12
|
11 | let uni: Universe = serde_json::from_str(&raw).unwrap();
| ---- `raw` is borrowed here
12 | return Some(uni);
| ^^^^^^^^^ returns a value referencing data owned by the current function
这里有什么问题吗?
uni.player_name
是一个包含 &str
的 Vec
。这个&str
借用了(意思是:是对)raw
的引用——但是raw
在get_config_from_file
末尾超出了范围code>,所以 uni
的生命周期不能超过这个值。但是,您尝试返回 uni
,这会使其生命周期比 get_config_from_file
的末尾更长——Rust 编译器会检测到此问题,并通过无法编译您的代码来防止它发生。
解决此问题的最简单的解决方案是不使用 &str
,而是使用 String
。由于这意味着 'a
生命周期现在未使用,因此您还需要将其删除。另外,您需要删除 #[serde(borrow)]
– 重点是代码不应再借用,因此该属性现在是错误的。
代码现在看起来像这样(并且可以编译):
use serde::{Deserialize};
#[derive(Debug, Deserialize)]
pub struct Universe {
player_name: Vec<String>,
}
pub fn get_config_from_file () -> Option<Universe> {
let raw: String = "some_data".to_string();
let uni: Universe = serde_json::from_str(&raw).unwrap();
return Some(uni);
}
关于rust - 如何从 String.split().collect() 返回数据?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/68668545/