SoFunction
Updated on 2025-04-03

In-depth understanding of TypeScript type compatibility

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 membersfeatAttributes, 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!