【问题标题】:Mysterious borrow scope extension神秘的借用范围扩展
【发布时间】:2015-08-22 03:22:15
【问题描述】:

为什么编译器会拒绝这段代码:

struct S<'a> {
    i: i32,
    r: &'a i32,
}

fn main() {
    let mut s = S{i: 0, r: &0};
    {
        let m1 = &mut s;
        m1.r = &m1.i;
    }
    let m2 = &mut s;
}

错误是:“不能一次多次借用 s 作为可变变量”(第一次借用:m1,第二次借用:m2)。

为什么s 的第一次借用在m1 超出范围后仍然存在?

我读到了超出原始借款人范围的借用范围扩展。然而,这似乎总是涉及“接管”原始借款的原始借款人范围之外的另一个借款人,例如此代码失败并出现完全相同的错误,这对我来说很清楚:

fn main() {
    let mut s = 0;
    let r: &mut i32;
    {
        let m1 = &mut s;
        r = m1;
    }
    let m2 = &mut s;
}

在第一个示例中,如果我将 m1.r = &amp;m1.i; 替换为 m1.r = &amp;dummy;(虚拟定义为一些 &i32)或 let dummy = &amp;m1.i;,则代码编译。仅当我将对某个字段的引用存储在借用结构的另一个字段中时,才会发生该错误。我不明白为什么这应该将借用扩展到其范围之外。

我对代码有什么问题的最佳猜测是:

  • s.r的原始生命周期是整个main

  • 当我分配对 m1.r 的引用时,它必须是原始生命周期,但 &amp;m1.i 仅在 m1 生命周期内有效。

但我可能是错的(错误信息会误导人)。

【问题讨论】:

    标签: rust


    【解决方案1】:

    首先要注意

    let mut s = S{i: 0, r: &0};
    {
        s.r = &s.i;
    }
    let m2 = &mut s;
    

    给予

    cannot borrow `s` as mutable because `s.i` is also borrowed as immutable
    

    希望这应该很清楚 - 如果结构自借,则它被借用。这指出了为什么任何自借结构基本上都是无用的——它不能被移动(使自己的指针无效),也不能对其进行可变引用。


    接下来需要了解,来自可变引用的不可变引用算作对可变引用的借用,所以扩展它。例如

    let mut v = ();
    let r1 = &(&mut v);
    let r2 = &v;
    

    给予

    cannot borrow `v` as immutable because it is also borrowed as mutable
    

    目前尚不清楚这是否可以在法律上成为对原始结构的新借用,但它目前还没有这样做。

    【讨论】:

    • @Levans,谢谢你的回答,这对我来说很清楚。有趣的部分实际上是嵌套范围:在我的示例中,借款人 m1 超出了范围,在您的示例中,尽管有花括号,“self”-borrower 仍然在范围内。只有在您的情况下,这才使我清楚错误。
    • “有趣的部分实际上是嵌套范围:在我的示例中,借用者 m1 超出范围” → 为什么假设 borrow 的范围与借款人的?借用者范围的唯一重要方面是它不能大于借用的范围,并且任何借用(例如&amp;mut &amp;mut T)都受到它的限制。它持有的借款的生命周期是无限的 - 考虑let x: &amp;'static str = "foo"
    • 你为什么认为我会这样认为?我试图通过使用术语借用范围而不是生命周期来明确说明,例如解释。 here。当然,借用范围可以大于借用者的范围,但前提是在第一个借用者的范围之外还有另一个借用者。
    • 但不是在这种情况下:m1m2 之前的唯一借款人。如果您注释行m1.r = &amp;m1.i;,您可以看到m1 的借用在右大括号处结束:代码编译。在m2 之前没有其他东西借用s
    • @dacker "但不是在这种情况下:m1m2 之前的唯一借款人。" → 不,m1.r = &amp;m1.i; 导致s 借用m1 的借用,因此s 可变地从s 借用。这显然会持续s 的生命周期。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2018-01-03
    • 1970-01-01
    • 1970-01-01
    • 2013-05-06
    • 1970-01-01
    • 2012-04-04
    相关资源
    最近更新 更多