typescript elegantly enforce a constraint while preserving types

前端 未结 2 624
Happy的楠姐
Happy的楠姐 2021-01-26 08:47

so originally to enforce a constraint, i\'d just apply an interface like this

// this is the constraint
interface Topic {
  [key: string]: (...args: any[]) =>          


        
2条回答
  •  终归单人心
    2021-01-26 09:25

    The way to deal with this is not to annotate your variable at all, but let the compiler infer its type. Then, sometime later, when you use that variable somewhere that expects a Topic, you'll get the desired error there:

    const topic = {
        async actuate(a: boolean) { }
    };
    topic.actuate(true);
    
    const badTopic = {
        oops(a: boolean) { }
    }
    badTopic.oops(true);
    
    // later ...
    
    function topicTaker(t: Topic) {
        // do something with a Topic
    }
    
    topicTaker(topic); // okay
    
    topicTaker(badTopic); // error!
    // ------> ~~~~~~~~
    // Property 'oops' is incompatible with index signature.
    

    So this works, and in some use cases this is really sufficient: you don't actually care if the variable is of the right type until you try to use it.

    But what if this error check is too far from the object declaration to be helpful to you?


    Well, if you want to catch the error earlier, you can do what I usually do and make a helper function:

    const asTopic = (topic: T) => topic;
    

    This constrained generic identity function just returns its argument at runtime. But at compile time it will require that topic is assignable to Topic without actually widening it to topic. Instead of annotating your variables, just initialize them to the return value of the helper function:

    const topic = asTopic({
        async actuate(a: boolean) { }
    });
    
    topic.actuate(true); // okay
    
    const badTopic = asTopic({
        oops(a: boolean) { } // error!
    //  ~~~~
    // void is not assignable to type Promise
    })
    

    You can look at asTopic() two ways: as a replacement for annotation to avoid widening (i.e., use asTopic() instead of : Topic), or as an early warning that your object is not of the type you want (i.e., asTopic() acts like a topicTaker() that you call immediately at the creation site instead of deferring it).

    Re-reading the question: this asTopic() helper function is probably the thing you were feeling around for with your TopicConstraint.

    Playground link to code

提交回复
热议问题