【发布时间】:2018-11-29 15:44:01
【问题描述】:
Mech 带有一个驱动程序,它是一个Named 实体。在运行时,省略的 Mech 构造函数会咨询外部源以获取要使用的特定类型的驱动程序。
trait Named {
fn name(self) -> String;
}
struct Person {
first_name: String,
last_name: String
}
impl Named for Person {
fn name(self) -> String {
format!("{} {}", self.first_name, self.last_name)
}
}
pub struct Mech<'a> {
driver: Box<Named + 'a>,
}
impl<'a> Mech<'a> {
pub fn driver_name(self) -> String {
self.driver.name()
}
}
方法driver_name 将所有权返回给String,以便在链接调用中进一步使用(在实际代码中它是Command)。编译失败:
error[E0161]: cannot move a value of type Named + 'a: the size of Named + 'a cannot be statically determined
--> src/lib.rs:22:9
|
22 | self.driver.name()
| ^^^^^^^^^^^
使特征Sized 使对象安全失败:
trait Named: Sized {
fn name(self) -> String;
}
↓
error[E0038]: the trait `Named` cannot be made into an object
--> src/lib.rs:17:5
|
17 | driver: Box<Named + 'a>,
| ^^^^^^^^^^^^^^^^^^^^^^^ the trait `Named` cannot be made into an object
|
= note: the trait cannot require that `Self : Sized`
有没有办法让这种模式发生?
我似乎缺少什么基本的东西吗?
如果这无法实现,有什么好的解决方法?
【问题讨论】:
-
为什么是
self而不是&self? -
除了采用 take 模式之外想不出其他方法:
fn take_name(&mut self) -> String是对象安全的,但需要对类型的状态域进行扩展。
标签: rust traits ownership trait-objects