我有一些类似于下面的代码,它尝试从 websocket 中读取数据,将 JSON 结果解析为一个结构,然后将该结构推送到 Vec
缓冲。然而,代码无法编译,因为结构有生命周期,并且借用检查器提示 JSON 字符串的生命周期不够长。
use serde::{Deserialize, Serialize};
use tungstenite::client::AutoStream;
use tungstenite::protocol::WebSocket;
#[derive(Serialize, Deserialize, Debug, Clone)]
struct MyType<'a> {
id: &'a str,
count: i64,
}
fn example<'a>(
conn: &mut WebSocket<AutoStream>,
buff: &'a mut Vec<MyType<'a>>,
) -> Option<Box<dyn std::error::Error>> {
match conn.read_message() {
Err(err) => Some(err.into()),
Ok(msg) => {
let resp_raw = msg.to_string();
let resp_parsed: Result<MyType<'a>, _> = serde_json::from_str(&resp_raw);
match resp_parsed {
Err(err) => Some(err.into()),
Ok(resp) => {
buff.push(resp.clone());
None
}
}
}
}
}
确切的错误是
borrowed value [&resp_raw] does not live long enough
.我想知道我应该如何重构这段代码以满足借用检查器;将具有生命周期的结构推送到
Vec
的正确方法是什么?参数?还是
&'a str
解析为 MyType
实际上仍然保留对原始 JSON 字符串的引用,所以没有办法安全地做到这一点?
最佳答案
看serde_json::from_str
小心:
pub fn from_str<'a, T>(s: &'a str) -> Result<T>
where
T: Deserialize<'a>,
这表示
T
反序列化与输入 s
共享相同的生命周期.这允许零拷贝反序列化,这就是你在 MyType
中得到的。 , 其中 id
是对字符串切片的引用。这将绑定(bind) MyType
的生命周期到 &resp_raw
的生命周期,这是 fn example()
的本地.这行不通。给
buff
不能解决问题你给它的生命周期参数。 example
-function 拥有 MyType
的缓冲区指向。允许MyType
“逃”到Vec
将允许创建悬空引用,因为缓冲区被销毁一次 example
返回。更改
MyType
满足DeserializeOwned
,即不带生命周期参数。你需要一个 String
或者一个(为了安全起见)一个Box<str>
而不是 &str
.
关于rust - 在 Rust 中,如何将具有生命周期的对象推送到向量中?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59116374/