A summary of some of the places where I spent time on TypeScript

A summary of some of the places where I spent time on TypeScript

Record some of the places where you spent time on ts.

(First of all, let me complain: stackoverflow really has everything, but Baidu is really useless)

pit

Compatibility misunderstanding of as assertion, such as "a" as "b" will not report an error.

Inconsistent behavior between interface and type (when I first encountered it, I thought I wrote the wrong type and was confused):

type Type = {
  key: "value"
}
interface Interface {
  key: "value"
}

type seems to be no different, both are true = Type extends Interface ? Type extends Interface ? true : false : false

type pit point = {
  [key: string]: pit point} | string

type test<T> = T extends pitfall? true : false
type This is true = Test <Type>
type This is false = Test <Interface>

There is an official explanation on GitHub that this pit was left intentionally. It is said that because the interface is extensible (the same name is automatically merged), it is inconvenient to detect.

When using generics to implement function overloading, in the implementation of the function, it is often necessary to use as forced assertions because generics do not have specific constraints.

//That's about it. I'm too lazy to actually test the following code 🙃

//fns is a function index table, TFns is the const type of the index table. Function overloading failed <T extends keyof TFns>(fn:T, params: Parameters<fns[T]>){
    fns[fn](...params) //In the implementation, the union type will not be narrowed, so an error will be reported //The error should be something like "cannot pass the parameters of method 1 to method 2"}
//But when used externally, it doesn't matter if the semantics of the type are met

The spread operator is not intuitive: [...string[], number] is intuitive to use (requiring a number element at the end of the array), but [...string[], null, ...object[], number] is not intuitive and will not report an error. The new version of ts adds a rule prohibiting continuous deconstruction, so this type of writing is not allowed.

Actually, there is a solution here, but the type written is simply unreadable (dozens of lines, including a large number of extends acting as type if judgments), so I will not post it. The code is posted below:

// Required types: [...number[], "middle-element", ...boolean[]] 
//The above is invalid, it just indicates what the following type code is used for (to implement the type constraints indicated above)

type Elem = number | boolean | "middle-element";

type Last<T extends any[]> = T extends [infer _]
  ? never
  : T extends [...infer _, infer Tl]
  ? Tl
  : never

type HandleEmpty<T extends any[], Data> = T['length'] extends 0 ? never : Data

type Validation<Params extends any[], Cache extends Elem[] = []> =
  Params extends []
  ? Cache['length'] extends 0
  ? never
  : Cache
  : Params extends [infer Fst, ...infer Rest]
  ? Cache extends []
  ? Fst extends number
  ? HandleEmpty<Rest, Validation<Rest, [...Cache, Fst]>>
  : never
  : Fst extends number
  ? Last<Cache> extends number
  ? HandleEmpty<Rest, Validation<Rest, [...Cache, Fst]>>
  : never
  : Fst extends "middle-element"
  ? Last<Cache> extends number
  ? HandleEmpty<Rest, Validation<Rest, [...Cache, Fst]>>
  : never
  : "middle-element" extends Cache[number]
  ? Fst extends boolean
  ? Validation<Rest, [...Cache, Fst]>
  : never
  : never
  : never

type IsNever<T> = [T] extends [never] ? true : false;

function check<
  Params extends Elem[],
  IsValid extends Validation<Params>
>(...arr: IsNever<IsValid> extends true ? [never] : [...Params]) {
  return arr
}

const normal = check(1, 'middle-element', false)
const error = check(false, "middle-element", 2)

Advanced Operations

Object name remapping:

//{ "new-a":any; "new-b":any }
type remap = {
    [K in "a" | "b" as `new-${K}`]: any
}

Splitting of union types: The infer keyword can be used to split union types.

//"a1"|"b2"
type split successfully <_Keys = keyof { a: 1, b: 2 }> = _Keys extends infer K ?
    `${Extract<K, string>}${{ a: 1, b: 2 }[Extract<K, _Keys>]}`
    : never

// Note: (as of ts4.4.4) Directly `keyof Obj extends infer K` cannot split the union type, the reason is unknown (too lazy to check 😁).
//The result is "a1"|"a2"|"b1"|"b2"
type split failed = keyof { a: 1, b: 2 } extends infer K ?
    `${Extract<K, string>}${{ a: 1, b: 2 }[Extract<K, "a" | "b">]}`
    : never

Tuple types:

  • The actual (non-type) parameters sometimes need to be explicitly declared as tuple types using as const.
  • Tuple types can get the exact length instead of the number through tuple["length"].
  • When a tuple type is used through a generic parameter, it is sometimes necessary to add []| and write it as a tuple extends []|any[] to avoid being parsed as a normal indefinite-length array type.

Recursive types: Use ...infer More to implement recursion on array types.

type Converter<T> = T extends string ? "str" ​​: null
//It goes in as [string,number,string], and comes out as ["str",null,"str"]
type recursion<
        Input source extends any[],
        The internal type cache extends any[] = []
    > = input source extends [any, ...infer remaining elements] ?
        Recursion <remaining elements, [...internal type cache, converter <input source [0]>]>
        : Input Source

Odds and Ends

  • & can be used instead of extends for type. Interface has no other difference except that it can merge types with the same name.
  • ts has a rich set of built-in types, here are a few examples:
    • ReturnType<function type>, gets the type of the return value of the function type.
    • Uncapitalize <string>, lock the first letter of the input string type to lowercase (other options include first letter capitalization, all lowercase, and all uppercase).

It is recommended for beginners to go to the official website to read the documents.

After getting into ts, you can pay attention to the new features (gameplay) brought by version updates.

Summarize

This is the end of this article summarizing the places where I spent time on TypeScript. For more relevant content about where I spent time on TypeScript, please search for previous articles on 123WORDPRESS.COM or continue to browse the related articles below. I hope everyone will support 123WORDPRESS.COM in the future!

<<:  Detailed explanation of Tomcat core components and application architecture

>>:  HTML tags list and usage instructions

Recommend

Detailed explanation of where the images pulled by docker are stored

The commands pulled by docker are stored in the /...

WeChat applet canvas implements signature function

In the WeChat applet project, the development mod...

Introduction to the role of HTML doctype

Document mode has the following two functions: 1. ...

CentOS7 uses rpm to install MySQL 5.7 tutorial diagram

1. Download 4 rpm packages mysql-community-client...

Detailed explanation of CSS background and border tag examples

1. CSS background tag 1. Set the background color...

Introduction to the usage of common XHTML tags

There are many tags in XHTML, but only a few are ...

Detailed explanation of MySQL's FreeList mechanism

1. Introduction After MySQL is started, BufferPoo...

Podman boots up the container automatically and compares it with Docker

Table of contents 1. Introduction to podman 2. Ad...

How to build lnmp environment in docker

Create a project directory mkdir php Create the f...

Example of pre-rendering method for Vue single page application

Table of contents Preface vue-cli 2.0 version vue...

Installation tutorial of the latest stable version of MySQL 5.7.17 under Linux

Install the latest stable version of MySQL on Lin...

Introduction to Linux compression and decompression commands

Table of contents Common compression formats: gz ...

How to use Tencent slider verification code in Vue3+Vue-cli4 project

Introduction: Compared with traditional image ver...