首页 > 解决方案 > 在类型别名上调用静态方法

问题描述

在使用盒装闭包时,我遇到了以下问题:

type Test = Rc<dyn Fn() -> i64>;

fn test_bad() -> Test {
    Test::new(|| 42)
}

fn test_good() -> Test {
    Rc::new(|| 42)
}

在第一种情况下,我使用类型别名来引用方法,而在第二种情况下new我直接使用。Rc

在第一种情况下,编译器抱怨:

    |       Test::new(|| 42)
    |             ^^^ function or associated item not found in `Rc<(dyn Fn() -> i64 + 'static)>`
    |
    = note: the method `new` exists but the following trait bounds were not satisfied:
            `dyn Fn() -> i64: Sized`

但第二种情况效果很好。有人可以解释一下区别吗?有什么方法可以new通过类型别名引用还是我需要自己包装它?

标签: typesrustclosurestype-aliastrait-objects

解决方案


test_good中,通过调用Rc::newon || 42,您不会创建一个Rc<dyn Fn() -> i64>,而是创建一个Rc<ClosureType>,其中提供的闭包ClosureType唯一类型是,它大小的。然后,由于在Rc<ClosureType>返回的函数中返回Rc<dyn Fn() -> i64>,因此它隐式地将其强制转换为特征对象。

test_bad失败是因为它没有从构造Rc一个大小的闭包然后将其转换为一个大小不一Rc的特征对象,而是尝试直接构造一个Rc大小不一的特征对象,这失败了,因为函数的所有参数都必须调整大小。

我认为没有直接的方法可以new通过类型别名进行引用,尽管您可以很容易地制作自己的:

fn new_test(func: impl 'static + Fn() -> i64) -> Test {
    Rc::new(func)
}

推荐阅读