首页 > 解决方案 > 循环中看似不一致的借用检查器行为

问题描述

在测试一个基本插件系统的实现时,我遇到了一些看似不一致的借用检查器行为。

具有以下结构

struct Plugin {
    data: [u8; 256],
}

impl Plugin {
    fn new() -> Plugin { Plugin { data: [0; 256], } }
    fn write(&mut self, port: usize, val: u8) { self.data[port] = val; }
    fn read(&self, port: usize) -> u8 { self.data[port] }
}

struct System<'a> {
    plugin: Option<&'a mut Plugin>,
}

impl<'a> System<'a> {
    fn new() -> System<'a> { System { plugin: None, } }
    fn attach_plugin(&mut self, plugin: &'a mut Plugin) { self.plugin = Some(plugin); }
    fn detach_plugin(&mut self) { self.plugin = None; }
}

给定以下设置代码

let mut system = System::new();
let mut plugin = Plugin::new();

system.attach_plugin(&mut plugin);

以下代码不起作用

for i in 0..255 {
    system.plugin.as_mut().unwrap().write(i as usize, i);
    let val = plugin.data[i as usize]; // This line produces two errors
    assert_eq!(val, i);
}

显示这两个错误消息:

error[E0503]: cannot use 'plugin.data' because it was mutably borrowed

error[E0503]: cannot use 'plugin.data[_]' because it was mutably borrowed

但是,如果通过系统的访问被分离到它自己的循环中,则代码编译时不会出现错误或警告。

for i in 0..255 {
    system.plugin.as_mut().unwrap().write(i as usize, i);
}
for i in 0..255 {
    let val = plugin.data[i as usize]; // This line doesn't produce any errors
    assert_eq!(val, i);
}

我注意到这两个示例之间唯一显着的区别是,在工作示例中,第一行在它自己的范围内,但是对产生错误的代码进行以下更改并不能修复错误:

for i in 0..255 {
    {
        system.plugin.as_mut().unwrap().write(i as usize, i);
    }
    let val = plugin.data[i as usize]; // This line still produces two errors
    assert_eq!(val, i);
}

因此,范围界定似乎并不是影响行为变化的因素。

为什么将代码分成两个循环会使代码停止产生错误?

标签: rustborrow-checker

解决方案


好吧,结构的生命周期在最后一个使用位置结束。

详细地说,您的原始代码的错误消息:

error[E0503]: cannot use `plugin.data` because it was mutably borrowed
  --> src/main.rs:29:15
   |
26 |     system.attach_plugin(&mut plugin);
   |                          ----------- borrow of `plugin` occurs here
27 |     for i in 0..255 {
28 |     system.plugin.as_mut().unwrap().write(i as usize, i);
   |     ------------- borrow later used here
29 |     let val = plugin.data[i as usize]; // This line produces two errors
   |               ^^^^^^^^^^^^^^^^^^^^^^^ use of borrowed `plugin`

它清楚地解释了system包含对plugin. 在 for 循环中,您首先使用,然后在下一行中system不可变地引用。pluginRust 不允许同时使用可变引用和不可变引用。请注意,您处于循环中,因此下一次迭代仍然使用,system即在执行时(至少在第一次迭代中)是活动的。systemlet val = plugin.data[i as usize]

如果将systemandplugin操作分成两个“for”循环它不会出错的原因是,在第一个“for”循环之后你不使用system,所以编译器决定它system的生命周期在第一个“for”之后立即结束“ 环形。这样就不会出现借用违规。

在一个循环中使用作用域没有帮助的原因仍然是,您还有循环的下一次迭代,所以system' 的生命周期不会在下一句之前结束。


推荐阅读