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) pitCompatibility 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: 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 OperationsObject 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:
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
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. SummarizeThis 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
The commands pulled by docker are stored in the /...
In the WeChat applet project, the development mod...
Document mode has the following two functions: 1. ...
1. Download 4 rpm packages mysql-community-client...
1. CSS background tag 1. Set the background color...
There are many tags in XHTML, but only a few are ...
1. Introduction After MySQL is started, BufferPoo...
Table of contents 1. Introduction to podman 2. Ad...
1. Vertical table and horizontal table Vertical t...
Create a project directory mkdir php Create the f...
Table of contents Preface vue-cli 2.0 version vue...
Install the latest stable version of MySQL on Lin...
Table of contents Common compression formats: gz ...
Introduction: Compared with traditional image ver...
Preface In MySQL, multi-table join query is a ver...