我有代码创建一个RefCell然后想要将它的引用传递RefCell给单个线程:
extern crate crossbeam;
use std::cell::RefCell;
fn main() {
let val = RefCell::new(1);
crossbeam::scope(|scope| {
scope.spawn(|| *val.borrow());
});
}
Run Code Online (Sandbox Code Playgroud)
在完整的代码中,我使用的是一种RefCell嵌入其中的类型(a typed_arena::Arena).我正在使用crossbeam来确保线程不会超过它所引用的引用.
这会产生错误:
error: the trait bound `std::cell::RefCell<i32>: std::marker::Sync` is not satisfied [E0277]
scope.spawn(|| *val.borrow());
^~~~~
Run Code Online (Sandbox Code Playgroud)
我相信我理解为什么会发生这样的错误:RefCell不是设计为从多个线程同时调用,并且因为它使用内部可变性,所以需要单个可变借位的正常机制不会阻止多个并发操作.这甚至记录在Sync:
不属于类型
Sync是那些具有"室内易变性"在非线程安全的方式,比如Cell和RefCell中std::cell.
这一切都很好,但在这种情况下,我知道只有一个线程能够访问RefCell.我怎样才能向编译器确认我理解我在做什么,我确保是这样的?当然,如果我认为这实际上是安全的是不正确的,我会非常乐意被告知原因.
我使用bindgen为相机的C++库创建了一个Rust包装器,并且C++库中的相机句柄定义为typedef void camera_handlebindgen移植的:
pub type camera_handle = ::std::os::raw::c_void;
Run Code Online (Sandbox Code Playgroud)
我能够成功连接到相机并拍摄图像,但是我想在单独的线程上运行代码来控制相机的温度,本质上是根据相机的当前温度改变冷却器功率,我想要与其余代码分开运行。这些调用需要相机句柄,但是当我生成新线程时,我不断收到错误:
pub type camera_handle = ::std::os::raw::c_void;
Run Code Online (Sandbox Code Playgroud)
在它的下面,它提到:
'*mut std::ffi::c_void' cannot be sent between threads safely
Run Code Online (Sandbox Code Playgroud)
如何将其发送到另一个线程,以便我也可以在那里使用该相机手柄?我尝试过使用脆弱和send_wrapper,但两者都没有成功。