rust - 将闭包传递给特征方法: expected type parameter,发现闭包
Posted 跨链技术践行者
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了rust - 将闭包传递给特征方法: expected type parameter,发现闭包相关的知识,希望对你有一定的参考价值。
我对如何使它正常工作有些困惑,我已经从真实的东西中删除了它。我写了一个特质:
pub trait Renderable<F: Fn(&PropertyTags)> {
fn set_property_changed_callback(&mut self, callback: Option<F>);
}
add_child
的'child'参数受哪个限制,而PropertyTags
只是一个枚举。我包括了child
类型的模拟实现,以演示我的用法:
pub struct Child<F: Fn(&PropertyTags)> {
property_changed_callback: Option<F>,
}
impl<F: Fn(&PropertyTags)> Renderable<F> for Child<F> {
fn set_property_changed_callback(&mut self, callback: Option<F>) {
self.property_changed_callback = callback;
}
}
然后将这些用作:
pub fn add_child<REND, C>(&mut self, child: &mut REND)
where C: Fn(&PropertyTags),
REND: Renderable<C>
{
let tc = Some(|property_tag: &PropertyTags|{
});
child.set_property_changed_callback(tc);
}
我收到错误消息:
child.set_property_changed_callback(tc);
| ^^ expected type parameter, found closure
|
= note: expected type `std::option::Option<C>`
= note: found type `std::option::Option<[closure@src/rendering/mod.rs:74:31: 76:18]>`
= help: here are some functions which might fulfill your needs:
- .take()
- .unwrap()
我设置了一个最小的游乐场示例,在此处重现了问题:https://play.rust-lang.org/?gist=bcc8d67f25ac620fe062032d8737954b&version=stable&backtrace=0
最佳答案
问题在于add_child
声称可以接受任何Renderable<C>
,其中C
可以是实现Fn(&PropertyTags)
的任何类型,但是该函数尝试为它提供一个特定的闭包类型,该类型可能与C
不同。
为了使其正常工作,add_child
的签名应如下所示:
pub fn add_child<REND>(&mut self, child: &mut REND)
where REND: Renderable<AddChildCallback>
其中AddChildCallback
是具体类型的名称(实现Fn(&PropertyTags)
)。
这里的困难在于,一方面,闭包类型没有可在Rust代码中使用的名称,另一方面,手动实现Fn
是不稳定的,因此需要每晚进行编译。
我还将注意到,通过将回调类型设置为类型参数,在设置了第一个回调之后,不能为Renderable
分配其他类型的回调,因为第一个回调将确定Renderable
的具体类型。这可能适合您的用法,我只是想确保您知道这一点。
如果您想要一个可以在稳定的编译器上运行的解决方案(从Rust 1.14.0开始),那么您就必须对回调进行包装。 add_child
的签名将如下所示:
pub fn add_child<REND>(&mut self, child: &mut REND)
where REND: Renderable<Box<Fn(&PropertyTags)>>
Here是更新的游乐场链接,其中包含Fn
的示例实现。请注意,根据trait定义的要求,call
,call_mut
和call_once
的参数作为元组传递。为了完整起见,以下代码被复制:
struct RenderableCallback {
}
impl<'a> Fn<(&'a PropertyTags,)> for RenderableCallback {
extern "rust-call" fn call(&self, args: (&'a PropertyTags,)) -> Self::Output {
}
}
impl<'a> FnMut<(&'a PropertyTags,)> for RenderableCallback {
extern "rust-call" fn call_mut(&mut self, args: (&'a PropertyTags,)) -> Self::Output {
}
}
impl<'a> FnOnce<(&'a PropertyTags,)> for RenderableCallback {
type Output = ();
extern "rust-call" fn call_once(self, args: (&'a PropertyTags,)) -> Self::Output {
}
}
以上是关于rust - 将闭包传递给特征方法: expected type parameter,发现闭包的主要内容,如果未能解决你的问题,请参考以下文章
Rust闭包和Haskell lambda有什么区别? [关闭]
将带有变量的闭包传递给 Laravel 查询构建器中的 where 方法