这个playground project有我正在尝试编译的代码的简化版本 .
use std::sync::{Arc, Mutex, MutexGuard};
pub trait Runnable {
fn run(&mut self) -> Option<String>;
}
pub struct Value {}
impl Runnable for Value {
fn run(&mut self) -> Option<String> {
Some("Value".to_string())
}
}
pub struct RunList {
runnables: Vec<Arc<Mutex<Runnable>>>,
}
impl RunList {
pub fn run<R>(&mut self, index: usize, mut runner: R)
where
R: FnMut(&mut MutexGuard<Runnable>),
{
let runnable_arc = self.runnables[index].clone();
let mut runnable = runnable_arc.lock().unwrap();
runner(&mut runnable);
}
}
fn main() {
let mut runnables = Vec::<Arc<Mutex<Runnable>>>::with_capacity(1);
runnables.push(Arc::new(Mutex::new(Value {})));
let mut run_list = RunList { runnables };
run_list.run(0, |runnable| {
println!("Hello, {}", runnable.run().unwrap());
});
}
我想要一个特征对象的向量,其中每个对象都受 Arc
和 Mutex
的保护,然后能够在每个对象上调用特征方法 .
我有"borrowed value does not live long enough"错误,但我看不出与this question/answer的区别 .
error[E0597]: `runnable_arc` does not live long enough
--> src/main.rs:25:28
|
25 | let mut runnable = runnable_arc.lock().unwrap();
| ^^^^^^^^^^^^ borrowed value does not live long enough
26 | runner(&mut runnable);
27 | }
| - borrowed value only lives until here
|
= note: borrowed value must be valid for the static lifetime...
2 回答
您似乎在编译器的推理系统中遇到了一个小问题 . 将
MutexGuard<Runnable>
更改为MutexGuard<Runnable + 'static>
可修复错误:但有趣的是,将其更改为
MutexGuard<'static, Runnable + 'static>
会保留错误:为什么编译器会推断后者而不是前者呢?那么,让我们来看看MutexGuard的定义:
在
MutexGuard<Runnable>
中,Runnable
是一个特征对象并且获得了一个隐式+ 'static
绑定,因为没有明确的生命周期 . 似乎编译器试图统一两次出现'a
,因为如果我们需要用'static
替换第二次出现,那么我们需要对第一次出现做同样的事情 . 但是,该逻辑会忘记一个重要的细节:'static
超过任何生命周期'a
,因此没有理由强制第一个'a
为'static
. 实际上,当我们写MutexGuard<Runnable + 'static>
时,它与MutexGuard<'a, Runnable + 'static>
相同(如何定义'a
取决于上下文) . 这是有效的,因为Runnable + 'static: 'a
(您总是可以将具有生命周期'x
的对象作为参数传递给期望生命周期更短的函数'y
) .为了记录,该问题与
FnMut
特征无关;正常函数也显示此行为:您正在将
MutexGuard
的引用传递给runner
,而不是对包装的runnable的引用 . 你需要插入一个deref:根据我的理解,这个构造 -
&mut *
- 在这里不应该是必要的,因为MutexGuard
实现了DerefMut
特征 . 显然,这是一个已知问题:Why does a mutable borrow of a closure through DerefMut not work?