我有一个简单的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();
});
Run Code Online (Sandbox Code Playgroud)
我该如何分配filename给path?如果我只是写
path = filename;
Run Code Online (Sandbox Code Playgroud)
我收到此错误:
src\main.rs:46:13: 46:28 error: cannot assign to captured outer variable in an `Fn` closure
src\main.rs:46 path = filename;
Run Code Online (Sandbox Code Playgroud)
出于几个原因你不能这样做.这是connect_clicked()方法的定义:
fn connect_clicked<F: Fn(Button) + 'static>(&self, f: F) -> u64
Run Code Online (Sandbox Code Playgroud)
该方法接受的闭包Fn是受限的'static.这意味着,首先,它不能修改它的环境(那将是FnMut),它也不能通过引用捕获任何东西(大致'static意味着什么).因此,有没有办法为关闭修改path像你想要的变量.
鉴于gtk-rs 强制说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);
Run Code Online (Sandbox Code Playgroud)
它也可以使用Arc<Mutex<..>>,但我不认为这是必要的,因为gtk-rs可以防止从主线程以外的任何线程使用GTK.
| 归档时间: |
|
| 查看次数: |
1163 次 |
| 最近记录: |