如何将变量移出闭包?

时间:2016-01-05 11:17:16

标签: closures rust

我有一个简单的GTK应用程序,我需要选择一个文件夹并将其传递给闭包之外的变量。

let mut path = "".to_owned();

button_open.connect_clicked(move |_| {
    let file_chooser = gtk::FileChooserDialog::new(
        "Open File", None, gtk::FileChooserAction::SelectFolder,
        [("Open", gtk::ResponseType::Ok), ("Cancel", gtk::ResponseType::Cancel)]);
    if file_chooser.run() == gtk::ResponseType::Ok as i32 {
        let filename = file_chooser.get_current_folder().unwrap();
    }

    file_chooser.destroy();
});

如何将filename分配给path?如果我只是写

path = filename;

我收到此错误:

src\main.rs:46:13: 46:28 error: cannot assign to captured outer variable in an `Fn` closure
src\main.rs:46             path = filename;

1 个答案:

答案 0 :(得分:8)

由于多种原因,您无法做到这一点。 Hereconnect_clicked()方法的定义:

fn connect_clicked<F: Fn(Button) + 'static>(&self, f: F) -> u64

此方法接受的闭包是Fn,并由'static限定。这意味着,首先,它无法修改其环境(即FnMut),也无法通过引用捕获任何内容(大致是'static的含义)。因此,闭包有 no 方式来修改path变量。

鉴于gtk-rs enforces GTK只能在主线程中使用,而且它的小部件不能Send能够使用,处理程序无法从另一个线程访问变量,所以我不确定为什么这些闭包有'static限制,我也完全没有理由认为它们不是FnMut。这似乎是一个实施缺陷。

无论如何,您可以使用Rc<RefCell<..>>创建可变数据:

let path: Rc<RefCell<Option<String>>> = Rc::new(RefCell::new(None));
let captured_path = path.clone();  // clone the Arc to use in closure
...
// inside the closure
*captured_path.borrow_mut() = Some(filename);

也可以使用Arc<Mutex<..>>,但我认为没有必要,因为gtk-rs可以防止从主线程以外的任何线程使用GTK。