为实现特征的所有类型实现特征

时间:2019-07-30 15:10:42

标签: generics rust traits

我有这个问题:

  • 实现特征Event的多个结构
  • 所有人都可以以相同的方式实现PartialEq特征

我考虑写这篇(简短版)

type Data = Vec<u8>;

trait Event {
    fn data(&self) -> &Data;
}

struct NoteOn {
    data: Data,
}
struct NoteOff {
    data: Data,
}
impl Event for NoteOn {
    fn data(&self) -> &Data {
        &self.data
    }
}
impl Event for NoteOff {
    fn data(&self) -> &Data {
        &self.data
    }
}
impl<T: Event> PartialEq for T {
    fn eq(&self, b: &T) -> bool {
        self.data() == b.data()
    }
}

fn main() {
    println!("Hello, world!");
}

playground

无法编译:

error[E0119]: conflicting implementations of trait `std::cmp::PartialEq<&_>` for type `&_`:
  --> src/main.rs:23:1
   |
23 | impl<T: Event> PartialEq for T {
   | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
   |
   = note: conflicting implementation in crate `core`:
           - impl<A, B> std::cmp::PartialEq<&B> for &A
             where A: std::cmp::PartialEq<B>, A: ?Sized, B: ?Sized;
   = note: downstream crates may implement trait `Event` for type `&_`

error[E0210]: type parameter `T` must be used as the type parameter for some local type (e.g., `MyStruct<T>`)
  --> src/main.rs:23:1
   |
23 | impl<T: Event> PartialEq for T {
   | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ type parameter `T` must be used as the type parameter for some local type
   |
   = note: only traits defined in the current crate can be implemented for a type parameter

这是怎么了? 还是有另一种方式来通用实现此PartialEq而不必为NoteOn键入一次,而为Noteff键入一次?

谢谢

1 个答案:

答案 0 :(得分:0)

这是我最好的“尝试”方式,它可以按照我的意愿进行操作,但以不同于@trentcl和@Shepmaster的方式进行。

我使用了@trentctl提到的枚举,以便能够在不同类型之间进行切换, 我将枚举值保留在通用的struct Event中,以便可以轻松包装不同的对象并向Event添加更多代码。 这样做还有助于简化Vec类型。

在这里,我想我只需要枚举,而不需要Event和带有枚举的属性,我仍在学习变体枚举用法

我还遇到了@trentcl提到的问题,即我不拥有Vec类型,因此我将Vec封装在struct Sec中,而不是简单地别名。 这使我的PartialEq的实现与Vec类型分开(如果我理解,但我不确定) =>我在这里遇到麻烦的原因是,我认为使用type A = B;会创建一个新类型,但是文档确实指出了它的别名(这可能会使我无法为Vec实现PartialEq) 尽管最后我想我可能对此也很错,因为我创建了一个只包装1个属性的人工结构这一事实似乎也适得其反

=>因此,现在总结一下,谢谢大家,我将继续努力,以了解如何提高效率,但是我想我只是在Rust的上下文中使用了错误的东西,不确定这是一个好答案,将不胜感激反馈或其他建议

https://play.rust-lang.org/?version=stable&mode=debug&edition=2018&gist=d67b7d993fa6b6285962ee58e9b215e5

type Data = Vec<u8>;

#[derive(Debug)]
enum EventImpl{
    NoteOn(u8),
    NoteOff(u8)
}
#[derive(Debug)]
struct Event{
    data:Data,
    i:EventImpl
}

impl Event{
    fn new(data:Data)->Self{
        Event{
            i: match data[0]{
                0 => EventImpl::NoteOn(data[1]),
                1 => EventImpl::NoteOff(data[1]),
                _ => panic!("unk")
            },
            data:data
        }
    }

    fn data(&self)->&Data{
        &self.data
    }
}
#[derive(Debug)]
struct Seq{
    pub things:Vec<Event>
}

impl PartialEq for Seq{
    fn eq(&self,o:&Self)->bool{
    // i have to learn the iterator implementation    
        let mut r=o.things.len()==self.things.len();
        if ! r{
            return false;
        }
        for i in 0..o.things.len() {
            r = r && o.things[i]==self.things[i];
        }
        r
    }
}
impl PartialEq for Event{
    fn eq(&self,o:&Self)->bool{
    // i have to learn the iterator implementation    
        std::mem::discriminant(&self.i) == std::mem::discriminant(&o.i) && o.data()==self.data()
    }
}


fn main() {
    let mut s:Seq=Seq{things:vec![Event::new(vec![1,2,3])]};
    s.things.push(Event::new(vec![0,1,2]));
    let s2:Seq=Seq{things:vec![Event::new(vec![1,2,3]),Event::new(vec![0,1,2])]};

    println!("Hello, world! {:?} {:?}",s, s.things[1].data());
    println!("S1 == S2 ? {}",s==s2);
}