是否有可能为由都实现了特征的类型组成的任何元组自动实现特征?

Mat*_*nti 1 tuples variadic traits rust

假设我有一个

trait Happy {}
Run Code Online (Sandbox Code Playgroud)

我可以Happy为任何我想要的结构实现,例如:

struct Dog;
struct Cat;
struct Alligator;

impl Happy for Dog {}
impl Happy for Cat {}
impl Happy for Alligator {}
Run Code Online (Sandbox Code Playgroud)

现在,我想自动为所有由所有实现该特征的类型组成的元组使用implHappyHappy特征。从直觉上讲,所有快乐的元组也是快乐的。

可以做这样的事情吗?例如,我可以简单地将实现扩展Happy到两种Happy类型的元组:

impl <T, Q> Happy for (T, Q) where T: Happy, Q: Happy {}
Run Code Online (Sandbox Code Playgroud)

结果,这可以完美地编译:

fn f(_: impl Happy) {
}

fn main() {
    f((Dog{}, Alligator{}));
}
Run Code Online (Sandbox Code Playgroud)

但是我怎么能将其推广到任何长度的元组呢?据我所知,Rust中没有可变参数的泛型。有解决方法吗?

She*_*ter 6

我们在Rust中没有可变参数的泛型。

正确。

有解决方法吗?

您使用宏:

trait Happy {}

macro_rules! tuple_impls {
    ( $head:ident, $( $tail:ident, )* ) => {
        impl<$head, $( $tail ),*> Happy for ($head, $( $tail ),*)
        where
            $head: Happy,
            $( $tail: Happy ),*
        {
            // interesting delegation here, as needed
        }

        tuple_impls!($( $tail, )*);
    };

    () => {};
}

tuple_impls!(A, B, C, D, E, F, G, H, I, J,);
Run Code Online (Sandbox Code Playgroud)

现在可以编译:

fn example<T: Happy>() {}

fn call<A: Happy, B: Happy>() {
    example::<(A, B)>();
} 
Run Code Online (Sandbox Code Playgroud)

通常,这不是一个大问题,因为长元组基本上是不可读的,如果确实需要,您可以始终嵌套元组。

也可以看看: