TypeScript union types, intersection types and type guards

TypeScript union types, intersection types and type guards

1. Union Type

The so-called union type is to define some types. The defined variables only need to satisfy any one type. The union type is defined using |. The sample code is as follows:

// Define the union type through the | symbol let value: number | boolean | string = '一碗周'
value = 18

In the above code we define a value variable, which can be a number, boolean or string type.

2. Crossover Type

We introduce union types, and then we look at intersection types, which are very similar to union types.

The so-called cross type needs to satisfy all types, and the cross type is defined using the & symbol.

The sample code is as follows:

// Define three common interface types interface Name {
  name: string
}
interface Age {
  age: number
}
interface Hobby {
  hobby: string
}
// Define an object that is a union type of the above three objects let person: Name & Age & Hobby = {
  // If one type is not allocated, an exception will be thrown name: '一碗周',
  age: 18,
  hobby: 'coding',
}

3. Type protection

Now we have a requirement: get the first number in an array of any type.

The implementation code is as follows:

// Define an array containing number or string const arr: (number | string)[] = [1, 'number']
// Traverse the array and return the first number const getValue: (arr: (number | string)[]) => number = (
  arr: (number | string)[],
): number => {
  for (let i = 0; i < arr.length; i++) {
    // If the current value is not a NaN when converted to a number, return if (!isNaN(Number(arr[i]))) {
      return arr[i] // Type 'string | number' cannot be assigned to type 'number'.
    }
  }
}


In the above code, when return , it is not known whether the returned value is of type number . So an exception will be thrown.

The above functions can be accomplished through type assertions. The sample code is as follows:

const getValue: (arr: (number | string)[]) => number = (
  arr: (number | string)[],
): number => {
  for (let i = 0; i < arr.length; i++) {
    // If the current value is not a NaN when converted to a number, return if (!isNaN(Number(arr[i]))) {
      return arr[i] as number // Tell the compiler that I am returning a number
    }
  }
}

What is a type assertion? Please refer to: Type assertion

If you use type assertions to explain, it will be more cumbersome if the desired data type is different. At this time, type protection is needed to complete this function.

There are three main types of type protection:

3.1 Custom Type Protection

The way to define custom type protection is to define a function whose return structure is in the form of parameterName is type , which is a type predicate. parameterName must be a parameter name from the current function parameters.

The sample code is as follows:

// Use custom type protection // 1. Define a function whose return value is a type predicate, that is, parameterName is Type, which is the form of parameter name is type function isNumber(value: number | string): value is number {
  // If true is returned, it means the value passed in is the type after is
  return !isNaN(Number(value))
}
// 2. Define a function to get a number const getNumber: (value: number | string) => number = (
  value: number | string,
): number => {
  // If the value of calling isNumber is true, it means value is a number, so return the number if (isNumber(value)) {
    return value
  }
}
// 3. Call to get the final value const getValue: (arr: (number | string)[]) => number = (
  arr: (number | string)[],
): number => {
  for (let i = 0; i < arr.length; i++) {
    // If a number is returned, convert it to a boolean value of true
    if (getNumber(arr[i]) || getNumber(arr[i]) === 0) {
      return getNumber(arr[i])
    }
  }
}

The reason for defining the second function is that there are still problems in directly using i as the return value in the array, so a function is defined as a transition.

3.2typeof type protection

The typeof keyword in JavaScript can determine the current type, but it can only determine four types: number , string , boolean , and symbol .

This is enough for this requirement. Next, let's see how to implement type protection through typeof .

The sample code is as follows:

// 1. Define a function to get a number const getNumber: (value: number | string) => number = (
  value: number | string,
): number => {
  // Check if the current value is a string, if so return the current value if (typeof value === 'number') {
    return value
  }
}
// 2. Call to get the final value const getValue: (arr: (number | string)[]) => number = (
  arr: (number | string)[],
): number => {
  for (let i = 0; i < arr.length; i++) {
    // If a number is returned, convert it to a boolean value of true
    if (getNumber(arr[i]) || getNumber(arr[i]) === 0) {
      return getNumber(arr[i])
    }
  }
}

3.3instanceof type protection

instanceof operator is also a native operator provided in JavaScript . It is used to determine whether an instance is created by a constructor or a class using ES6 syntax. In TypeScript, type protection can also be achieved through instanceof operator.

The sample code is as follows:

/**
 * Since instanceof only supports reference types and not primitive types, we need to make some changes here and modify the array to the following:
 */
const arr2: (Number | String)[] = [new String('彼岸繁华'), new Number(10)]
// 1. Define a function to get a number const getNumber: (value) => number = (value): number => {
  // Determine whether the current value is of Number type and convert the current value into a string and return if (value instanceof Number) {
    return Number(value)
  }
}
// 2. Call to get the final value const getValue: (arr: (Number | String)[]) => number = (
  arr: (Number | String)[],
): number => {
  for (let i = 0; i < arr.length; i++) {
    // If a number is returned, convert it to a boolean value of true
    if (getNumber(arr[i]) || getNumber(arr[i]) === 0) {
      return getNumber(arr[i])
    }
  }
}


There are two points to note when using instanceof:

  • Only applicable to any reference type, not primitive types.
  • Whether the prototype chain of the former contains the prototype object of the latter.

This concludes this article on TypeScript union types, intersection types, and type protection. For more information about TypeScript union types, intersection types, and type protection, please search previous articles on 123WORDPRESS.COM or continue browsing the following related articles. I hope you will support 123WORDPRESS.COM in the future!

You may also be interested in:
  • TypeScript union types, intersection types and type guards
  • Detailed explanation of TypeScript 2.0 marked union types
  • A brief discussion on TypeScript's type protection mechanism

<<:  Basic usage of @Font-face and how to make it compatible with all browsers

>>:  Use HTML to write a simple email template

Recommend

Solution to Nginx SSL certificate configuration error

1. Introduction When a web project is published o...

Analysis of MySQL cumulative aggregation principle and usage examples

This article uses examples to illustrate the prin...

Whitespace processing in HTML/CSS and how to preserve whitespace in the page

Whitespace rules in HTML In HTML, multiple spaces...

HTML+Sass implements HambergurMenu (hamburger menu)

A few days ago, I watched a video of a foreign gu...

Quickly solve the problem that CentOS cannot access the Internet in VMware

Yesterday I installed CentOS7 under VMware. I wan...

Detailed explanation of concat related functions in MySQL

1. concat() function Function: Concatenate multip...

Summary of the advantages of Vue3 vs. Vue2

Table of contents 1. Why do we need vue3? 2. Adva...

vue-cli4.5.x quickly builds a project

1. Install vue-cli npm i @vue/cli -g 2. Create a ...

Summary of the dockerfile-maven-plugin usage guide

Table of contents pom configuration Setting.xml c...

How to design and optimize MySQL indexes

Table of contents What is an index? Leftmost pref...

Vue implements 3 ways to switch tabs and switch to maintain data status

3 ways to implement tab switching in Vue 1. v-sho...

Code analysis of synchronous and asynchronous setState issues in React

React originated as an internal project at Facebo...

Introduction to the use of MySQL source command

Table of contents Thoughts triggered by an online...