将相同变量绑定到具有特征的不同类型的模式

问题描述 投票:0回答:1

我有一个关于通过特征共享某些行为的值的模式匹配的问题。

我有一个带有两个变体的枚举,每个变体的绑定值都不同,其中两种类型都实现一个特征。我试图弄清楚是否有可能创建单个模式(E::VarA(x) | E::VarB(x)形式),在该模式中,我将两种类型都绑定到单个常量,只要我只对使用共享行为感兴趣。

一个示例:Playground

trait T {
    fn f(&self) -> usize;
}

struct A;

impl T for A {
    fn f(&self) -> usize { 1 }
}

struct B;

impl T for B {
    fn f(&self) -> usize { 2 }
}

enum E {
    VarA(A),
    VarB(B),
}

fn unwrap(e: E) -> usize {
    match e {
        E::VarA(v) | E::VarB(v) => T::f(&v)
    }
}

fn main() {
    let val = E::VarA(A{});  
    println!("{}", unwrap(val));
}

该代码显然不会编译,但是显示了我的意图。有没有一种方法可以使代码工作,最好比简单地将pat1 | pat2 => ...拆分为pat1 => ... ; pat2 => ...更优雅?

rust pattern-matching traits
1个回答
0
投票

您可以创建一个可解开以匹配语句的宏。

trait T {
    fn f(&self) -> usize;
}

struct A;
impl T for A {
    fn f(&self) -> usize { 1 }
}

struct B;
impl T for B {
    fn f(&self) -> usize { 2 }
}

enum E {
    VarA(A),
    VarB(B),
}

macro_rules! unwrap {
    ($value:expr, $pattern:pat => $result:expr) => {
        match $value {
            E::VarA($pattern) => $result,
            E::VarB($pattern) => $result,
        }
    };
}

fn main() {
    let a = E::VarA(A{});
    let b = E::VarB(B{});

    println!("a:{} b:{}",
        unwrap!(a, ref sm => sm.f()),
        unwrap!(b, ref sm => sm.f()));

}
© www.soinside.com 2019 - 2024. All rights reserved.