Generic Constraints
If you remember from an earlier example, you may sometimes want to write a generic function that works on a set of types where you have some knowledge about what capabilities that set of types will have. In our loggingIdentity
example, we wanted to be able to access the .length
property of arg
, but the compiler could not prove that every type had a .length
property, so it warns us that we can’t make this assumption.
1 2 3 4 | function loggingIdentity<T>(arg: T): T { console.log(arg.length); // Error: T doesn't have .length return arg; } |
Instead of working with any and all types, we’d like to constrain this function to work with any and all types that also have the .length
property. As long as the type has this member, we’ll allow it, but it’s required to have at least this member. To do so, we must list our requirement as a constraint on what T can be.
To do so, we’ll create an interface that describes our constraint. Here, we’ll create an interface that has a single .length
property and then we’ll use this interface and the extends
keyword to denote our constraint:
1 2 3 4 5 6 7 8 | interface Lengthwise { length: number; } function loggingIdentity<T extends Lengthwise>(arg: T): T { console.log(arg.length); // Now we know it has a .length property, so no more error return arg; } |
Because the generic function is now constrained, it will no longer work over any and all types:
1 | loggingIdentity(3); // Error, number doesn't have a .length property |
Instead, we need to pass in values whose type has all the required properties:
1 | loggingIdentity({length: 10, value: 3}); |
Please login to continue.