Curiously recurring generic trait pattern: overflow evaluating the requirement

前端 未结 2 1483
太阳男子
太阳男子 2021-01-26 14:03

I am trying to implement a generic structure with a bunch of fields, where each of the field types should know about the exact type of the whole structure. It\'s a sort of strat

相关标签:
2条回答
  • 2021-01-26 14:53

    First of all, everything becomes a lot clearer if you avoid putting trait bounds on definitions of structs and traits. When things get complicated, the constraints are at least solved from the same direction.

    pub struct Example<S, D, A> {
        pub s: S,
        pub a: A,
        pub data: D,
    }
    
    pub trait Strategy<T> {
        type Associated;
        fn run(&self, &T);
    }
    
    pub trait HasData {
        type Data;
        fn data(&self) -> &Self::Data;
    }
    
    impl<S, D, A> Example<S, D, A>
    where
        S: Strategy<Self, Associated = A>,
    {
        pub fn do_it(&self) {
            self.s.run(self);
        }
    }
    
    impl<S, D, A> HasData for Example<S, D, A>
    where
        S: Strategy<Self, Associated = A>,
    {
        type Data = D;
        fn data(&self) -> &D {
            &self.data
        }
    }
    

    Your implementation of Strategy for ExampleStrat looks like this:

    impl<E: HasData<Data = ExampleData>> Strategy<E> for ExampleStrat {
        type Associated = ();
         // ...
    }
    

    What this means is that you are defining it for all possible qualifying types E. The type-checker can now only look at the trait bounds, which are again generic and only expressed in terms of other traits, which use each other as bounds, so the type-checker gets into a cycle. Put a block in the cycle by giving it a concrete type, which you know.

    pub struct ExampleStrat;
    pub struct ExampleData;
    
    impl Strategy<Example<ExampleStrat, ExampleData, ()>> for ExampleStrat {
        type Associated = ();
        fn run(&self, e: &Example<ExampleStrat, ExampleData, ()>) {
            let _ = e.data();
            // uses ExampleData here
        }
    }
    
    fn main() {
        let example = Example {
            s: ExampleStrat,
            a: (),
            data: ExampleData,
        };
        example.do_it();
    }
    
    0 讨论(0)
  • 2021-01-26 15:07

    If the following impl is characteristic for Strategy, then it might be parameterized on the wrong thing. (I'm going to ignore the associated type for this answer, because the example doesn't use it.)

    impl<E: HasData<Data = ExampleData>> Strategy<E> for ExampleStrat {
        fn run(&self, e: &E) {
            let _ = e.data();
            // uses ExampleData here
        }
    }
    

    You could instead parameterize Strategy over D -- breaking the impl dependency cycle -- and parameterize only the run method over E.

    pub trait Strategy<D> {
        fn run(&self, &impl HasData<Data = D>);
    }
    
    impl Strategy<ExampleData> for ExampleStrat {
        fn run(&self, e: &impl HasData<Data = ExampleData>) {
            let _ = e.data();
            // uses ExampleData here
        }
    }
    

    fn run<E: HasData<Data = ExampleData>>(&self, e: &E) is another way to define run that is the same for this purpose. Here is a full example.

    A potential drawback of this approach is that run can't be called through a Strategy trait object, because it has to be monomorphized for any type that implements HasData. But the HasData trait doesn't seem to do much in this impl: the only thing it can do is return an internal reference, and once you have it, there's no point in using it again. Maybe run could just take a &D reference?

    pub trait Strategy<D> {
        fn run(&self, &D);
    }
    
    impl Strategy<ExampleData> for ExampleStrat {
        fn run(&self, _: &ExampleData) {
            // uses ExampleData here
        }
    }
    

    To be sure, now you have to call self.s.run(self.data()) in do_it, but this doesn't cost you in flexibility over the original version, in which, had it worked¹, you could only call Strategy<E>::run with an argument of type &E.

    In fact, the whole HasData trait seems unnecessary to me: it's always implemented by the same type whose implementation calls it, so aside from the minor convenience of passing self instead of self.data, it doesn't elevate the level of abstraction inside the do_it method. So it seems to me effectively the same thing to delete HasData entirely and let Example know how to call Strategy::run with the right reference; it has to, anyway. (However, it's possible I merely lack imagination.)

    Any of these solutions ought to handle adding an associated type to Strategy, but without knowing how it will be used, it's hard to say for sure.

    ¹It could be made to work in some future version of the compiler, with sufficiently smart type checking.

    0 讨论(0)
提交回复
热议问题