gpt4 book ai didi

struct - 试图将自引用数据拆分成一个单独的结构

转载 作者:行者123 更新时间:2023-11-29 08:08:03 26 4
gpt4 key购买 nike

我希望能够在 Parent 中存储一个名为 Child 的结构,其中 Child 包含对父级的引用。

如果我将 Child 结构直接放在父级内部,它就可以工作,如下所示:

struct Parent<'s> {
cache: RefCell<Vec<Child<'s>>>
}

但是如果我将 Vec 移动到一个单独的结构中,那么它将无法编译并出现生命周期错误。

struct Parent<'s> {
cache: RefCell<Cache<'s>>
}

struct Cache<'s> {
children: Vec<Child<'s>>
}

可以使这个示例与单独的结构一起工作吗?

这是 full working code ,编译得很好。当将 children 移动到单独的结构中时,它会 fails .

我对问题的分析:

Parent 直接包含 children 时,Parent 结构的范围具有相同的生命周期本身,因此我可以在 Parent 上调用采用 &'s self 的方法。

Parent 包含 CacheCache 包含 children 时, 的生命周期与范围相同Cache 结构,它在Parent 之前创建,因此无法调用Parent 上的方法,该方法采用&'s self .尝试这样做会给出错误

<anon>:33:15: 33:16 error: `p` does not live long enough
<anon>:33 let obj = p.create_object();
^
<anon>:30:48: 38:2 note: reference must be valid for the block suffix following statement 0 at 30:47...
<anon>:30 let cache = Cache { children: Vec::new() }; // the lifetime `'s` is essentially from this line to the end of the program
<anon>:31 let mut p = Parent { cache: RefCell::new(cache) }; // although the Parent instance was created here, 's still refers to the lifetime before it
<anon>:32 // this fails because p doesn't live long enough
<anon>:33 let obj = p.create_object();

我需要一种方法将 缩短到 Parent 的范围,而不是 Cache 的范围。

免责声明:这个问题与我之前提出的问题(https://stackoverflow.com/questions/32579518/rust-lifetime-error-with-self-referencing-struct?noredirect=1#comment53014063_32579518)非常相似,被标记为重复。我已经通读了答案,我相信我可以做到这一点,因为我可以获得正确的引用生命周期(如我的第一个示例所示)。我再次问这个(现在略有不同)问题,因为我现在有一个有效的具体示例,一个无效的示例。我确信一个结构可以完成的事情可以用两个来完成,对吧?

最佳答案

您可以通过在相同的 let 绑定(bind)中定义 CacheParent 具有相同的生命周期来使其编译。

fn main() {
let (cache, mut p);
cache = Cache { children: Vec::new() };
p = Parent { cache: RefCell::new(cache) };
let obj = p.create_object();

let c1 = Child { parent: &p, data: 1 };
p.cache.borrow_mut().children.push(c1);
}

在这里,我们实际上是在声明一个解构的元组,然后对其进行初始化。我们不能直接在 let 绑定(bind)上初始化元组:

    let (cache, mut p) = (Cache { children: Vec::new() }, Parent { cache: RefCell::new(cache) });

因为 p 的初始化器引用了 cache,但是直到 let 语句结束时才定义该名称。单独的初始化之所以有效,是因为编译器跟踪初始化了哪些变量;如果你交换分配的顺序,你会得到一个编译器错误:

<anon>:31:38: 31:43 error: use of possibly uninitialized variable: `cache` [E0381]
<anon>:31 p = Parent { cache: RefCell::new(cache) };

关于struct - 试图将自引用数据拆分成一个单独的结构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32674398/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com