When we learn type compatibility, we are learning whether TypeScript can assign values to other types. This section will introduce TypeScript's type compatibility rules in functions, enums, classes, and generics in detail.
1. Explanation
Type compatibility is used to determine whether a type can be assigned to other types.
TypeScript's type checking mechanism is designed to allow developers to intuitively discover code writing problems during the compilation stage, develop good code specifications, and avoid many low-level errors.
let address: string = 'Baker Street 221B' let year: number = 2010 address = year // Error
Code explanation:Line 3, type ‘number’ cannot be assigned to type ‘string’.
2. Structured
TypeScript type compatibility is based on structure types; structure types use only their members to describe types.
The basic rule of TypeScript structured type system is that if x is to be y compatible, then y has at least the same properties as x . for example:
interface User { name: string, year: number } let protagonist = { name: 'Sherlock·Holmes', year: 1854, address: 'Baker Street 221B' } let user: User = protagonist // OK
Code explanation:Each attribute in the interface User can find the corresponding attribute in the proteinist object and the type matches. In addition, you can see that the proteinist has an additional property address, but the assignment will also succeed.
3. Comparison of two functions
Relatively speaking, it is easier to understand when comparing primitive types and object types. The difficult thing is how to determine whether the two functions are compatible.To determine whether the two functions are compatible, the first thing to do is to see whether the parameters are compatible, and the second thing to do is to see whether the return value is compatible.
3.1 Function parameters
Let's take a look at a code example:
let fn1 = (a: number, b: string) => {} let fn2 = (c: number, d: string, e: boolean) => {} fn2 = fn1 // OK fn1 = fn2 // Error
Code explanation:
In line 4, assigning fn1 to fn2 is true because:
- Each parameter of fn1 can find the corresponding type of parameter in fn2
- The parameter order remains the same, the parameter type corresponds to the
- Parameter names do not need to be the same
In line 5, assigning fn2 to fn1 does not hold true because the required parameters in fn2 must find the corresponding parameters in fn1. Obviously, the third Boolean parameter is not found in fn1.
The parameter types are just the same, and it does not need to be exactly the same:
let fn1 = (a: number | string, b: string) => {} let fn2 = (c: number, d: string, e: boolean) => {} fn2 = fn1 // OK
Code explanation:The first parameter of fn1 is the joint type of number and string, which can correspond to the first parameter type of fn2, so the assignment on line 4 is normal.
3.2 Function returns value
Create two functions that are only of different return value types
Code explanation:In the last line, the function x() lacks the location attribute, so an error is reported.
The type system forces the return value type of the source function to be a subtype of the return value type of the target function.. From this we can find that if the return value type of the objective function is void, the return value of the source function can be of any type:
let x : () => void let y = () => 'mybj' x = y // OK
4. Type compatibility of enumerations
Enumerations are compatible with numeric types:
enum Status { Pending, Resolved, Rejected } let current = let num = 0 current = num num = current
Different enum types are incompatible:
enum Status { Pending, Resolved, Rejected } enum Color { Red, Blue, Green } let current = current = // Error
5. Type compatibility of classes
The compatibility of classes with object literals and interfaces is very similar, but the classes are divided into instance parts and static parts.
When comparing data of two class types, only instance members will be compared, and static members and constructors will not be compared.
class Animal { feet!: number constructor(name: string, numFeet: number) { } } class Size { feet!: number constructor(numFeet: number) { } } let a: Animal let s: Size a = s! // OK s = a // OK
Code explanation:Class Animal and class Size have the same instance membersfeat
Attributes, and the types are the same. Although the constructor parameters are different, the constructor does not participate in the comparison of the two class types, so the last two lines can be assigned to each other.
Private and protected members of the class affect compatibility.Subclasses are allowed to assign values to the parent class, but cannot assign values to other classes of the same type.
class Animal { protected feet!: number constructor(name: string, numFeet: number) { } } class Dog extends Animal {} let a: Animal let d: Dog a = d! // OK d = a // OK class Size { feet!: number constructor(numFeet: number) { } } let s: Size a = s! // Error
Code explanation:
Line 13, the subclass can be assigned to the parent class.
Line 14: The reason why the parent class can be assigned to a subclass is that there are no members in the subclass.
The last line, because the member feet in class Animal are protected, the assignment cannot be successful.
6. Type compatibility of generics
The type compatibility of a generic varies depending on whether it is used by a member. Let's take a look at a code example:
interface Empty<T> {} let x: Empty<number> let y: Empty<string> x = y! // OK
In the above code, x and y are compatible because their structures are not different when using type parameters. But when generics are used by members:
interface NotEmpty<T> { data: T } let x: NotEmpty<number> let y: NotEmpty<string> x = y! // Error
Code explanation:Because in line 4, the generic parameter is of type number and in line 5, the generic parameter is of type string, the assignment of the last line fails.
If no generic parameters specifying generic type are specified, all generic parameters will be compared as any type
7. Summary
We must make full use of TypeScript's type checking mechanism to regulate the code and reduce some unnecessary errors. This is also our original intention of using TypeScript.
This is the end of this article about in-depth understanding of TypeScript type compatibility. For more related TypeScript type compatibility content, please search for my previous articles or continue browsing the related articles below. I hope everyone will support me in the future!