在Rust中,当您需要一个引用保持结构有时拥有其引用的数据时,该模式是什么?
考虑以下数据结构:
struct ReferenceHoldingStruct<'a> {
pub prop: &'a str
}
这种模式通常很有用,特别是在解析器(我的用例)中,用于“给”一个结构一些字符串数据而不重新分配它:
fn generate_these_1<'a>(input: &'a str) -> ReferenceHoldingStruct<'a> {
ReferenceHoldingStruct { prop: input }
}
但是,我有一个例子,我有一个像上面这样的结构,但在一个地方我需要生成“独立”的实例;即他们拥有自己的数据:
fn generate_these_2<'a>(input: String) -> ReferenceHoldingStruct<'a> {
ReferenceHoldingStruct { prop: input.as_str() }
}
我理解为什么这个版本不起作用:被引用的 String 并不存在于 Rust 可以看到它会徘徊以继续满足结构的&str引用的任何地方。我想也许这会奏效:
fn generate_these_2<'a>(input: String) -> (String, ReferenceHoldingStruct<'a>) {
(input, ReferenceHoldingStruct { prop: input.as_str() })
}
因为至少字符串不会立即被删除。我想也许 Rust 可以弄清楚这个元组既包含引用又包含它引用的数据,因此只要它们像这样保持在一起,它们就是有效的。但是没有骰子:这仍然不起作用。它将引用视为借用,将向元组的移动视为移动,您不能“同时”执行这两项操作。
所以,我理解这个问题,但不知道从哪里开始。这种情况的标准做法是什么?
回答
这是一个示例Cow(在评论中提到):
use std::borrow::Cow;
struct ReferenceHoldingStruct<'a> {
pub prop: Cow<'a, str>
}
fn generate_these_1<'a>(input: &'a str) -> ReferenceHoldingStruct<'a> {
ReferenceHoldingStruct { prop: Cow::Borrowed(input) }
}
fn generate_these_2<'a>(input: String) -> ReferenceHoldingStruct<'a> {
ReferenceHoldingStruct { prop: Cow::Owned(input) }
}
回答
你可以存储一些实现的东西Borrow<str>:
struct ReferenceHoldingStruct<S: Borrow<str>> {
pub prop: S,
}
绑定S允许它是任何可以作为&str. 这可能是一个&str本身,或一个拥有的String(或任何其他实现Borrow<str>)。由于Borrow<T>在T和上都实现了全面的实现&T,因此这种方法可用于存储引用或拥有任何类型的值,而不仅仅是str.
操场
THE END
二维码