在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.

操场


以上是在Rust中,当您需要一个引用保持结构有时拥有其引用的数据时,该模式是什么?的全部内容。
THE END
分享
二维码
< <上一篇
下一篇>>