我有一个关于通过特征共享某些行为的值的模式匹配的问题。
我有一个带有两个变体的枚举,每个变体的绑定值都不同,其中两种类型都实现一个特征。我试图弄清楚是否有可能创建单个模式(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 => ...
更优雅?
您可以创建一个可解开以匹配语句的宏。
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()));
}