Author: Qiong Xin
preface
TypeScript is becoming increasingly popular with developers as part of a trend in front-end development. In general, almost 90% of frameworks and toollibraries are written with TypeScript (or similar type schemes such as Flow). On the smaller side, TypeScript is a great help even for writing configuration files (like vite’s) or small scripts (thanks TS-Node). It is impossible for everyone to like the same thing. For example, Remy Sharp, the author of Nodemon, once said that he had never used TS (see #1565) and would never learn it in the future, probably because of language habits. There’s another big hurdle that often discourages newcomers from picking up TypeScript: the cost of learning.
In the early days of TypeScript, many students had a love-hate relationship with TypeScript because of its typing hints and engineering capabilities. However, TypeScript was often plagued by type errors and ended up using any.
The main culprit is actually part of the problem students, at the beginning of the learning TypeScript or to revolt, in the case of a blank TS program, either don’t know how to learn, then the conscience of the official document, look at a few relevant article will feel, finally have a problem or a scratching his head.
This article addresses the latter by trying to focus on TypeScript’s type programming part (how many other parts does TS have? See below for an explanation), starting with the most basic generics, going through types like index, map, and condition to keywords like IS, in, and infer, and finally the final tool type. Open your IDE and follow me through the code to take your TypeScript to the next level.
It is important to note that this article is not intended as an introduction to TypeScript and is not intended for those who have no current experience with TypeScript. If you’re still new to TypeScript, I recommend the XcatLiu introduction to TypeScript tutorial and the official TypeScript documentation. From my experience, you can read the tutorial early and head to the official documentation if you’re confused.
Come back to this article once you’ve finished getting started with TypeScript.
TypeScript = Type programming + ES proposal
I usually break TypeScript into two parts:
-
Pre-implemented ES proposals such as decorators, optional chains? ., the null merge operator?? (introduced with optional chains in TypeScript3.7), private members of classes, and so on. With the exception of some extremely unstable syntax (I’m talking about you, decorators), most TS implementations are actually the future ES syntax.
Technically speaking, the ES version decorator and TS version decorator are two things now. I previously introduced some history of TS decorator in the article Approaching MidwayJS: First Introduction to TS Decorator and IoC Mechanism. Interested students may wish to read it.
For this part, whether you have previous experience with JavaScript, Java, or C#, you will be able to get started very quickly. After all, the main thing is syntax sugar. This, of course, is the most commonly used part of actual development, since it is more down to earth than the other part: type programming.
-
Type programming, from a simple interface, to the advanced looking T extends SomeType, to all kinds of obscure tool types, Partial, Required, and so on. This part has no effect on the actual functional level of the code, even if you have ten any lines of code, you will get type errors like @ts-ignore (similar to @eslint-ignore, which will disable type checking on the next line). Even turning on transpileOnly (which disables type checking in the TS compiler and only compiles code for faster compilation) doesn’t affect the logic of your code itself. However, this is why type programming has not been given much attention: compared to syntax, it brings a lot of extra code (type definition code may even exceed business code) and so on. The actual business does not need to be very demanding type definition, usually only interface data, application state flow, etc., usually the underlying framework library will need a lot of type programming code. If the previous section made your code sweeter, the most important function of this section is to make your code more elegant and robust (yes, elegant and robust are not incompatible). If your team uses a monitoring platform like Sentry, Cannot read property ‘XXX’ of undefined, undefined is not a function Although it’s unlikely that even TS will be able to erase the error completely, it’s likely that it will.
Ok, so with all that groundwork in place, it’s time to get down to business. The chapters in this article are distributed as follows. If you already have some of the basics (such as generics), you can skip them.
-
The foundation of type programming: generics
-
Type guard with the IS and IN keywords
-
Index type and mapping type
-
Condition type, distributed condition type
-
Infer the keyword
-
Tool type
-
New TypeScript 4.x features
The generic
I started with generics because it is the most fundamental part of TypeScript’s typing architecture, and all advanced types are written on it. Just as we cannot program without variables, variables in type programming are generics.
Suppose we have a function like this:
function foo(args: unknown): unknown { ... }
Copy the code
-
If it receives a string, return a partial intercept of that string.
-
If you receive a number, return n times that number.
-
If you receive an object, return the object with the changed key (the same key name).
What these scenarios have in common is that the return value of the function is of the same type as the input parameter.
What should I do if I want to get a precise type definition here?
-
The unknown replacement string | number | object? But what this means is that this function accepts any value, and its return type could be String/number/object.
Don’t forget that we want the effect to be the same as the return value type. This is where generics come in. We collect the type value of the parameter with a generic and return it as a value, like this:
function foo<T>(arg: T): T {
return arg;
}
Copy the code
So when we use the function foo, the editor can determine the return value of the function in real time based on the parameters we pass in. Just as in programming, the value of a variable in a program is determined at runtime, the value (type) of a generic type is determined when a method is called, a class is instantiated, and so on.
Generics make the type definition of a code fragment easy to reuse (such as a subsequent implementation of a function that receives Boolean and returns Boolean) and increase flexibility and rigor.
In addition, you may have seen the use of Array
Map
. Usually we use unassigned forms like T in the above example as type parameter variables or generic types. An instantiated Array
is called an actual type parameter or a parameterized type.
Typically, generics use only a single letter. T, U, K, V, S, etc. My recommendation is to use generic variable declarations with specific meaning, such as the form BasicBusinessType, once the project reaches a certain level of complexity.
foo<string>("linbudu");
const [count, setCount] = useState<number>(1);
Copy the code
The example above can also be left unspecified because TS automatically deduces the actual type of a generic type, and in some Lint rules, it’s actually not recommended to add a type value that can be automatically deduced.
Writing generics under arrow functions:
const foo = <T>(arg: T) => arg;
Copy the code
If you write this in a TSX file,
may be recognized as a JSX tag, so you need to explicitly tell the compiler:
const foo = <T extends SomeBasicType>(arg: T) => arg; Copy the code
In addition to being used in functions, generics can also be used in classes:
class Foo<T, U> { constructor(public arg1: T, public arg2: U) {} public method(): T { return this.arg1; }}Copy the code
So much for the introduction to generics alone (because there is nothing to be said about generics alone), we will cover more generics in the advanced Types section that follows.
Type guard, is in keyword
Let’s dive into generics based type programming, starting with a relatively straightforward topic: type guards.
Suppose we have a field that could be either a string or a number:
numOrStrProp: number | string;
Copy the code
Now, if you want to narrow the union type of this field to, say, string, you might write:
export const isString = (arg: unknown): boolean => typeof arg === "string";
Copy the code
Look at this:
function useIt(numOrStr: number | string) { if (isString(numOrStr)) { console.log(numOrStr.length); }}Copy the code
The isString function doesn’t seem to narrow down the type. The arguments are still associative types. This is the time to use the is keyword:
export const isString = (arg: unknown): arg is string =>
typeof arg === "string";
Copy the code
If isString(numOrStr) is true, then numOrStr is reduced to string. This is just a union type with primitive types as members, and we can extend this to various scenarios. Let’s start with a simple false value judgment:
export type Falsy = false | "" | 0 | null | undefined; export const isFalsy = (val: unknown): val is Falsy => ! val;Copy the code
This is probably one of the most common type aliases I use every day, along with type guards like isPrimitive and isFunction.
And use the in keyword, we can further narrow Type (Type Narrowing), think about the following example, how will A | B “joint Type to” A “?
class A { public a() {} public useA() { return "A"; } } class B { public b() {} public useB() { return "B"; }}Copy the code
First think of for… The in loop, which iterates over the name of an object’s property, and the in keyword, which determines whether a property is owned by the object:
function useIt(arg: A | B): void {
'a' in arg ? arg.useA() : arg.useB();
}
Copy the code
If there is a attribute in the argument, since the intersection of type A and type B does not contain a, this will immediately narrow the type to a.
Since the intersection of types A and B does not contain the attribute A, the in judgment here precisely Narrows the type pairs to before and after the ternary expression. That’s either A or B.
Here’s another example of using a literal type as a type guard:
interface IBoy { name: "mike"; gf: string; } interface IGirl { name: "sofia"; bf: string; } function getLover(child: IBoy | IGirl): string { if (child.name === "mike") { return child.gf; } else { return child.bf; }}Copy the code
About literal type literal types, it is the type of further restrictions, such as your status code can only be 0/1/2, and then you can write the status: 0 | 1 | 2 form, rather than use a number to express.
Literal types include string literals, numeric literals, Boolean literals, and the template literals introduced in 4.1 (which we’ll explore later).
Such as string literals, common mode: “dev” | “prod”.
Boolean literals are often blended with other literal type, such as the open: true | | “none” “chrome”.
This kind of basic knowledge will be interspersed in various parts of the article to avoid the lack of specific scenes and the monotonous concept.
Distinguish interfaces based on fields
In daily life, I often see students asking similar questions: the user information under login and login is completely different interface, or
You can also use the in keyword to solve this problem. You can also use the in keyword to solve this problem. You can also use the in keyword to solve this problem.
interface ILogInUserProps {
isLogin: boolean;
name: string;
}
interface IUnLoginUserProps {
isLogin: boolean;
from: string;
}
type UserProps = ILogInUserProps | IUnLoginUserProps;
function getUserInfo(user: ILogInUserProps | IUnLoginUserProps): string {
return 'name' in user ? user.name : user.from;
}
Copy the code
Or by literal types:
interface ICommonUserProps {
type: "common",
accountLevel: string
}
interface IVIPUserProps {
type: "vip";
vipLevel: string;
}
type UserProps = ICommonUserProps | IVIPUserProps;
function getUserInfo(user: ICommonUserProps | IVIPUserProps): string {
return user.type === "common" ? user.accountLevel : user.vipLevel;
}
Copy the code
In the same vein, you can also use instanceof to guard the type of an instance.
Index type and mapping type
The index type
Before you start reading this section, you need to be prepared for a shift in your thinking and to really realize that type programming is actually programming, because from here on out, we’ll really be using generics as variables to do all kinds of fancy things.
Just as you often iterate over an object when writing business code, we often iterate over an interface in type programming. So you can reuse some of your programming ideas. First, implement a simple function that returns an object’s key:
Function pickSingleValue(obj, key) {return obj[key]; }Copy the code
What do I need to define in order to type it?
-
Parameters of the obj
-
The key parameters
-
The return value
There is a correlation between these three:
-
Key must be one of the key names in obj, and must be of type string (normally we only use string as object key names).
-
The value returned must be the key in obj
Therefore, our preliminary results are as follows:
function pickSingleValue<T>(obj: T, key: keyof T) {
return obj[key];
}
Copy the code
Keyof is the syntax for an indexed query that returns a literal union type of the key value of the type parameter that follows. For example:
interface foo {
a: number;
b: string;
}
type A = keyof foo; // "a" | "b"
Copy the code
Is it like object.keys ()? The difference is that it returns the union type.
Joint Type Union Type usually use | syntax, on behalf of multiple possible values, is actually at the beginning, we had used. The main use case for union types is the conditional type section, which is covered in a full section later.
It’s missing the return value. If you haven’t seen this syntax before, you might get stuck. In syntax, we might write something like this while traversing an object:
const fooObj = { a: 1, b: "1" };
for (const key in fooObj) {
console.log(key);
console.log(fooObj[key]);
}
Copy the code
As we did above, if we get the key, we get the corresponding value, so the type of value is much simpler:
function pickSingleValue<T>(obj: T, key: keyof T): T[keyof T] {
return obj[key];
}
Copy the code
This part may not be easy to understand in one step, so let’s explain:
interface T { a: number; b: string; } type TKeys = keyof T; // "a" | "b" type PropAType = T["a"]; // number Copy the code
You can use the key name to retrieve the key value of the object, and you can also retrieve the key value (and type) of the interface
But there are obvious improvements to this writing: keyof appears twice, and the generic T should really be restricted to object types. For the first point, do what you would normally do in programming: use one variable to store multiple occurrences. Remember, in type programming, generics are variables.
function pickSingleValue<T extends object, U extends keyof T>(
obj: T,
key: U
): T[U] {
return obj[key];
}
Copy the code
Here’s another new thing: extends… What is it? You can temporarily think of T extends Object as T restricted to an object type, and U extends keyof T as a generic type. U must be an associative type of the key names of the generic type T (in the form of a literal type, such as T, the key names of the object include a, B, and C, So U value can only be “a” “b” “c”, one of the “a” | | “b” “c”). We’ll talk more about this in the chapter on Conditional types.
Let’s say that instead of just fetching a single value, we want to fetch a set of values, that is, argument 2 will be an array of the keys of argument 1:
function pick<T extends object, U extends keyof T>(obj: T, keys: U[]): T[U][] {
return keys.map((key) => obj[key]);
}
// pick(obj, ['a', 'b'])
Copy the code
There are two important changes:
-
Keys: U[] We know that U is an associative type of T keys, so we can use this method to represent an array whose internal elements are all T keys. See the distributed Conditional Types section below for details.
-
T [U] [] the principle of it, in fact, the same as above, the first is T [U], on behalf of the 1 Key parameters (like Object [Key]), I think it is a good example, shows the combination of TS type programming, you don’t feel this kind of writing is like building blocks?
Index Signature Index Signature
In JavaScript, we usually index arrays using arr[1] and objects using obj[key]. In plain English, an index is how you get the members of an object, and in type programming, an index signature is used to quickly create an interface with internal fields of the same type, such as
interface Foo {
[keys: string]: string;
}
Copy the code
Then interface Foo is actually equivalent to an interface with all string keys and no restrictions on members.
Equivalent to Record
,>
, see tool type.
It is important to note that the JS can access the object properties at the same time through the Numbers and strings, so keyof Foo would result in a string | number.
Const o: Foo = {1: "wuhu! }; o[1] === o["1"]; // trueCopy the code
But once an interface has an index signature of type number, the object using it can no longer be accessed through the string index, such as o[‘1’], and an error will be thrown. The element implicitly has type “any” because the index expression is not of type “number”.
Mapped Types
Before we start mapping types, let’s first think about the map method of arrays in JavaScript. By using map, we get a new array from an array according to a given mapping relationship. In type programming, we map a new type definition from a type definition (including but not limited to interfaces, type aliases). It is usually a modification of an old type, such as:
-
Example Change the key type of the original interface
-
Add modifiers to the original interface key type, such as readonly and optional?
Start with a simple scenario:
interface A {
a: boolean;
b: string;
c: number;
d: () => void;
}
Copy the code
Now we have A requirement to implement an interface whose fields are exactly the same as interface A, but all of them are of type String. What would you do? Just restate one and write it? That’s crazy. We’re smart programmers.
If you want to copy an object (assuming it is not nested, regardless of the address of the reference type variable), the common way to do this is to first create a new empty object, and then fill the new object by traversing the key and value pairs of the original object. The same goes for interfaces:
type StringifyA<T> = {
[K in keyof T]: string;
};
Copy the code
Familiar? The important thing is the in operator, which you can completely interpret as for… in/for… After we get the key name, the key value is simple, so we can easily copy a new type alias.
type ClonedA<T> = {
[K in keyof T]: T[K];
};
Copy the code
With this in mind, you are already exposed to the underlying implementation of some of the tool types:
You can think of tool types as public functions that you normally put in the utils folder, providing a wrapper around common logic (in this case, type-programming logic), such as the two type interfaces above. For more information on tool types, see the tool Types chapter.
Let’s start with the most common Partial taste, which we’ll cover in a special section:
Type Partial<T> = {[K in keyof T]? : T[k]; };Copy the code
key? : value for this field is optional, in most cases is equal to the key: value | is undefined.
Conditional Types
In programming, we often use If statements with ternary expressions, but I personally prefer the latter, even If:
if (condition) {
execute()
}
Copy the code
I also like to write this If statement without an else:
condition ? execute() : void 0;
Copy the code
The syntax for conditional types, which are essentially ternary expressions, is the simplest example:
T extends U ? X : Y
Copy the code
If you think extends here is a little confusing, you can just temporarily understand that the properties in U are all in T.
Why are there conditional types? You can see that conditional types are often used in conjunction with generics, and given the use scenarios of generics and the latency of inference, I think you get the idea. For scenarios where the type cannot be determined in real time, use conditional types to dynamically determine the final type at run time (run time may not be accurate, or it can be understood that when you provide a function to be used by someone else, you dynamically determine the type constraints that need to be satisfied based on the parameters passed in when the other person uses it).
Similar to a programming statement, the value of a variable is dynamically assigned based on a condition:
let unknownVar: string; unknownVar = condition ? "Tao department front end" : "Taobao FED"; type LiteralType<T> = T extends string ? "foo" : "bar";Copy the code
Condition type understand actually also very intuitive, the only need to have a certain understanding of the cost of the type system is when conditions will gather enough information to determine the type, that is to say, sometimes not immediately complete type judgment, such as tool library provides functions, requires the user to type are not complete in the incoming parameters when using conditions of judgment.
Before we get to that point, let’s look at a common scenario for conditional types: a generic constraint, which is actually our index type example above:
function pickSingleValue<T extends object, U extends keyof T>(
obj: T,
key: U
): T[U] {
return obj[key];
}
Copy the code
The T extends Object and U extends keyof T are both generic constraints that restrict T to the object type and U to the union type of literals with T key names (remember? Tip: 1 | 2 | 3). We usually use generic constraints to narrow the type constraints. Simply put, generics themselves are open to all types. All types can be passed explicitly (for example Array
) or implicitly derived (for example foo(1)), which is not what we want, just as we sometimes check the arguments to functions:
function checkArgFirst(arg){
if(typeof arg !== "number"){
throw new Error("arg must be number type!")
}
}
Copy the code
In TS, we through generic constraint, requires the incoming generics can only be fixed type, such as T extends {} constraint generics to object types, T extends number | string will be generic constraint to type Numbers and strings.
Let’s take an example of using a conditional type as the return value type of a function:
declare function strOrNum<T extends boolean>(
x: T
): T extends true ? string : number;
Copy the code
In this case, the derivation of the conditional type is delayed because the type system does not have enough information to complete the judgment.
The derivation can be completed only if the required information (in this case, the type of the input parameter x) is given.
const strReturnType = strOrNum(true);
const numReturnType = strOrNum(false);
Copy the code
Similarly, just as ternary expressions can be nested, conditional types can also be nested. If you look at the source code of some frameworks, you will find that there are many nested conditional types.
type TypeName<T> = T extends string
? "string"
: T extends number
? "number"
: T extends boolean
? "boolean"
: T extends undefined
? "undefined"
: T extends Function
? "function"
: "object";
Copy the code
Distributive Conditional Types
A distributed condition type is actually not a special condition type, but one of its properties (so it is more accurate to say that a condition type is distributed). For check types that are parameters of the bare type, conditional types are automatically distributed to the union type at instantiation time.
The original:
Conditional types in which the checked type is a naked type parameter are called distributive conditional types. Distributive conditional types are automatically distributed over union types during instantiation
Let’s extract a few key words, and then clarify the concept through examples:
-
Bare type parameters (Type parameters are generics, as described in the generics section at the beginning of this article)
-
instantiation
-
Distribute to the union type
// Use the TypeName alias above
// “string” | “function” type T1 = TypeName<string | (() => void)>;
// “string” | “object” type T2 = TypeName<string | string[]>;
// “object” type T3 = TypeName<string[] | number[]>;
We found that in the above example, the conditions in the derived results are joint types (T3 is actually, only be merged because the result is the same), and is actually a type parameter is, in turn, the condition judgment, the reuse | combination of results.
Did you get something? In the example above, the generics are naked. If they are wrapped, how does the conditional type judgment change? Here’s another example:
type Naked<T> = T extends boolean ? "Y" : "N";
type Wrapped<T> = [T] extends [boolean] ? "Y" : "N";
// "N" | "Y"
type Distributed = Naked<number | boolean>;
// "N"
type NotDistributed = Wrapped<number | boolean>;
Copy the code
-
Among them, the Distributed type alias, its type parameters (the number | Boolean) will be the right distribution, namely first Distributed to Naked < number > | Naked < Boolean >, and then to judge, so the result is “N” | “Y”.
-
And NotDistributed type alias, at first glance feel TS should automatically according to the distributed array, the result should be “N” | “Y”? But in fact, its type parameters (the number | Boolean) does not have a distribution process, directly [number | Boolean] extends Boolean judgment, so the result is “N”.
Now we can talk about these concepts:
-
A bare type parameter, which has no extra [] wrapped around it, like an array, can no longer be called a bare type parameter.
-
Instantiation, in fact, is the process of determining the condition type. As we said earlier, the condition type needs to collect enough inferential information before it can proceed. The instantiation process for the two examples is actually different, as described in the next section.
-
Distribution to the union type:
-
For TypeName, its internal type parameter T is not wrapped, so TypeName < string | () = > (void) > will be distributed to TypeName < string > | TypeName < () = > (void) >, Then judge again, and finally distributed to “string” | “function”.
-
Abstract concrete process:
( A | B | C ) extends T ? X: Y // is equivalent to (A extends T? X : Y) | (B extends T ? X : Y) | (B extends T ? X: Y) // No additional distribution logic will be applied after the [] package is used. [A | B | C] extends [T] ? X : YCopy the code
In a nutshell: the union type parameters that are not extra wrapped by [] will be distributed and judged separately when the conditional type is determined.
There is no good or bad between these two behaviors. The only difference is whether to distribute the federated type. If you need to distribute conditional types, be careful to keep your type parameters as raw type parameters. If you want to avoid this behavior, just wrap your type arguments around [] (note that both sides of the extends keyword are needed).
Infer the keyword
In the case of conditional types, we have shown how to defer type determination through conditional judgment, but using only conditional types has a downside: it doesn’t get the type information from the condition. For example, T extends Array
Such scenarios are common, such as getting the type of return value from a function and unbobbing a Promise/array, so infer is introduced in this section.
Infer is the abbreviation of inference. It is usually used to refine generic types as type parameters. For example, infer R, R indicates the type to be inferred. Usually infer is not used directly but is placed in the underlying tool type along with the condition type. If the condition type provides the ability to delay inference, infer can provide the ability to delay inference based on conditions.
Take a look at a simple example of the utility type ReturnType used to get the return value type of a function:
const foo = (): string => { return "linbudu"; }; type ReturnType<T> = T extends (... args: any[]) => infer R ? R : never; // string type FooReturnType = ReturnType<typeof foo>;Copy the code
-
(… Args: any[]) => infer R is a whole, where the function’s return value type is taken by infer R.
-
When ReturnType is called, the type parameters T and R are explicitly assigned (T is typeof foo, and infer R is assigned to string as a whole, which is the return value typeof the function). If T meets the constraint of the condition type, the infer value of R is returned. In this case, R is the actual type of the return value of the function.
-
In fact, for the sake of rigor, we should constrain the generic T to be a function type, namely:
// The first extends constraint can only be passed in as a function type. // The second extends extends extends as a condition. Type ReturnType<T extends (... args: any[]) => any> = T extends (... args: any[]) => infer R ? R : never;Copy the code
Infer may not be a good habit to use. We can use a common example in front-end development: when the page is initialized, display placeholder interaction, such as Loading/skeleton screen, and then render the real data after the request returns. Infer is also the same idea. The type system can deduce the type parameters following infer after acquiring enough information (usually from delayed inference of conditions) and usually return the inferred results.
Similarly, we can get the input types of functions, the input types of class constructors, and even the internal types of promises, which we’ll cover later.
Also, in the case of function overloads in TS, using infer (ReturnType above) does not perform a derivation for all overloads, only the last overload (because in general the last overload is usually the most extensive case) will be used.
Tool Type Tool Type
This chapter is probably the most “cost effective” part of the article, because even if you don’t know a lot about how these types of tools are implemented after reading this section, it doesn’t matter if you use them properly, just as Lodash doesn’t require you to know how every function you use works.
This section includes the types of tools built into TS and the types of extensions to the community. I personally recommend that after you finish your study, select some of the tools that you find valuable, that your existing or future business might use, or that you just find fun. Store it in a new.d.ts file (or /utils/tool-types.ts) in your project.
Before you continue reading, make sure you have the above information, which is the basis for the type of tool.
Types of built-in tools
Above we have implemented one of the most used types of built-in tools:
type Partial<T> = { [K in keyof T]? : T[k]; };Copy the code
It is used to make all fields in an interface optional, except for the index type and the mapping type. Optional modifier, so I’m just going to pull out my cheat sheet:
-
Remove optional modifiers: -? , location and? consistent
-
Read-only modifier: readonly, in the key name, such as readonly key: string
-
To remove the read-only modifier: -readonly, the position is the same as readonly.
Congratulations, you’ve got Required and Readonly (the utility type that removes the Readonly modifier is not built-in, as we’ll see later) :
type Required<T> = { [K in keyof T]-? : T[K]; }; type Readonly<T> = { readonly [K in keyof T]: T[K]; };Copy the code
Above we implement a pick function:
function pick<T extends object, U extends keyof T>(obj: T, keys: U[]): T[U][] {
return keys.map((key) => obj[key]);
}
Copy the code
Similarly, suppose we now need to select some fields from an interface:
type Pick<T, K extends keyof T> = { [P in K]: T[P]; }; / / / / expected usage expectation result A type (" A ") | A type (" b ") type Part = Pick < A, "A" | "b" >;Copy the code
It’s still the mapping type, except now the mapping source for the mapping type is the type parameter K passed to Pick.
With Pick, there’s a natural need to Omit it. It’s very similar to Omit, but there’s one more problem we need to solve: how do we mean the extra padding in T without K?
Pick selects incoming keys and Omit incoming keys
Here’s another point: the never type, which represents a type that will never occur, is often used to narrow down a union type or interface, or as a base for conditional typing. Detailed can see you big zhihu answer, here we do not expand the introduction.
The above scenario can be simplified as follows:
// "3" | "4" | "5"
type LeftFields = Exclude<"1" | "2" | "3" | "4" | "5", "1" | "2">;
Copy the code
Exclude means Exclude, so the first parameter should be the filter, and the second parameter should be the filter! Try this first:
In fact, we use the characteristic of distributed conditional type, assuming that Exclude accepts T and U, the types in T union type will be judged in turn against U, and if the type parameter is in U, it will be excluded (assign the value to never).
Grounding gas version: “1” in the “1” | “2” inside it (” 1 “extends” 1 “|” 2 “- > true)? If it is, remove it (set to never), if it is not, keep it.
type Exclude<T, U> = T extends U ? never : T;
Copy the code
Then it’s easy to Omit the original vomit members and use Pick instead.
type Omit<T, K extends keyof any> = Pick<T, Exclude<keyof T, K>>;
Copy the code
For example, Exclude removes the key name, and Exclude retains the key name. For example, Exclude excludes the key name.
type Extract<T, U> = T extends U ? T : never;
Copy the code
Record
is a common tool used to generate new interfaces with associative types (Keys) and key-value types (Type).
type MyNav = "a" | "b" | "b"; interface INavWidgets { widgets: string[]; title? : string; keepAlive? : boolean; } const router: Record<MyNav, INavWidgets> = { a: { widget: [""] }, b: { widget: [""] }, c: { widget: [""] }, };Copy the code
In fact, it is simple to take each key value of Keys and specify the Type as Type
Type Record<K extends keyof any, T> = {[P in K]: T; };Copy the code
Note that Record supports Record
as well as string extends keyof any. Because the end result of keyof must be a union type of string (except in the case of using a number as the key name…) .
In the infer section earlier we implemented the ReturnType used to get the return value of the function:
type ReturnType<T extends (... args: any) => any> = T extends ( ... args: any ) => infer R ? R : any;Copy the code
In fact, infer can be used to obtain Parameters by changing its position, such as putting it into a parameter:
type Parameters<T extends (... args: any) => any> = T extends ( ... args: infer P ) => any ? P : never;Copy the code
If you take the bolder step and replace the normal function with the class constructor, you get the ConstructorParameters that take the types of the class constructor’s parameters:
type ConstructorParameters< T extends new (... args: any) => any > = T extends new (... args: infer P) => any ? P : never;Copy the code
Add the new keyword to make it an instantiable type declaration, where the constraint generic is class.
This is the type of the constructor that gets the class. If you put the infer type where it returns, think about what the return value of a new class is. The instance! So we get the InstanceType InstanceType:
type InstanceType<T extends new (... args: any) => any> = T extends new ( ... args: any ) => infer R ? R : any;Copy the code
You should get the hang of it by looking at these examples. Type programming does not have a particularly sophisticated syntax. It is a test of how well you know the basic parts of it, such as indexes, mappings, conditional types, and how well you can draw inferences from one another. The types of community tools we’ll look at below are essentially a combination of basic types, starting with common scenarios and adding to what the authorities don’t cover.
Community Tool Types
Most of the utility types in this section come from utility-types, and the authors of the react-Redux-typescript – Guide and typesafe-Actions are also excellent.
At the same time, I also recommend the Library of Type-FEST, which is more grounded than above. The author’s work… I guarantee you have used it directly or indirectly (if you don’t believe me, be sure to check it out, I was really shocked when I first saw it).
We’ll start by encapsulating the basic type alias and the corresponding type guard:
export type Primitive = | string | number | bigint | boolean | symbol | null | undefined; export const isPrimitive = (val: unknown): val is Primitive => { if (val === null || val === undefined) { return true; } const typeDef = typeof val; const primitiveNonNullishTypes = [ "string", "number", "bigint", "boolean", "symbol", ]; return primitiveNonNullishTypes.indexOf(typeDef) ! = = 1; }; export type Nullish = null | undefined; export type NonUndefined<A> = A extends undefined ? never : A; This tool / / TS actually has a built-in type type NonNullable < T > = T extends null | undefined? never : T;Copy the code
Falsy and isFalsy we’ve already shown above.
While we can warm up the memory of infer, let’s take a look at a common scene to extract the actual type of Promise:
const foo = (): Promise<string> => {
return new Promise((resolve, reject) => {
resolve("linbudu");
});
};
// Promise<string>
type FooReturnType = ReturnType<typeof foo>;
// string
type NakedFooReturnType = PromiseType<FooReturnType>;
Copy the code
Infer is actually easy to write if you already know how to use infer. Just use a infer parameter as a generics of the Promise:
export type PromiseType<T extends Promise<any>> = T extends Promise<infer U>
? U
: never;
Copy the code
Infer R is used to wait for the type system to deduce the specific type of R.
Recursive tool types
We’ve written Partial Readonly Required and other utility types that decorate interface fields, but they all have limitations. What if there is nesting in the interface?
type Partial<T> = { [P in keyof T]? : T[P]; };Copy the code
Here’s the logic:
-
If it’s not an object type, it just adds? The modifier
-
If it is an object type, then iterate through the inside of the object
-
Repeat the process.
We’ve seen this a couple of times, T extends object, but how do you iterate inside an object? It’s actually recursion.
export type DeepPartial<T> = { [P in keyof T]? : T[P] extends object ? DeepPartial<T[P]> : T[P]; };Copy the code
The internal implementation of utility-types is actually more complicated than that, and also takes into account the array case, which is simplified here for the sake of understanding, as is the case with the later utility types.
Deep obly and Deep GUIDANCE would be simple:
export type DeepMutable<T> = { -readonly [P in keyof T]: T[P] extends object ? DeepMutable<T[P]> : T[P]; }; DeepReadonly export type DeepImmutable<T> = {+readonly [P in keyof T]: T[P] extends object? DeepImmutable<T[P]> : T[P]; }; export type DeepRequired<T> = { [P in keyof T]-? : T[P] extends object | undefined ? DeepRequired<T[P]> : T[P]; };Copy the code
Especially pay attention to the DeepRequired, its condition is T type judgment [P] extends object | undefined, because nested object types may be optional (undefined), if only use object, may lead to incorrect results.
Another way to save your mind is to recurse all attributes directly without judging the condition type
Returns the tool type of the key name
In some cases we need a tool type that returns a combination of interface field keys and then uses that combination for further manipulation (such as use of Pick or Omit), usually the key name will meet specific criteria, such as:
-
Optional/Mandatory/read-only/non-read-only fields
-
Field of (not) object/(not) function/type
Let’s take a look at the simplest function type field FunctionTypeKeys:
export type FunctTypeKeys<T extends object> = { [K in keyof T]-? : T[K] extends Function ? K : never; }[keyof T];Copy the code
{[K in keyof T]: … }[keyof T] [keyof T]
interface IWithFuncKeys { a: string; b: number; c: boolean; d: () => void; } type WTFIsThis<T extends object> = { [K in keyof T]-? : T[K] extends Function ? K : never; }; type UseIt1 = WTFIsThis<IWithFuncKeys>;Copy the code
It is easy to deduce that UseIt1 is in fact:
type UseIt1 = {
a: never;
b: never;
c: never;
d: "d";
};
Copy the code
UseIt reserves all fields. The key value of a field that meets the criteria is the literal type (key name). The key value of a field that does not meet the criteria is never.
Add the following part:
// "d"
type UseIt2 = UseIt1[keyof UseIt1];
Copy the code
The process is similar to permutation: values of type never do not appear in the union type
To get rid of / / never type is automatically string | number type WithNever = string | never | number;Copy the code
So {[K in keyof T]:… }[keyof T] returns the name of the key.
Non-functional fields are easy to use, so let’s take a look at the optional OptionalKeys field and the required dKeys field. Let’s look at a small example:
type WTFAMI1 = {} extends { prop: number } ? "Y" : "N"; type WTFAMI2 = {} extends { prop? : number } ? "Y" : "N";Copy the code
It’s easy to figure out the answer if you can get around it. In the case of the former, the prop is required, so an empty object {} does not satisfy extends {prop: number}, as it does in the case of the optional prop.
Therefore, we use this idea to get optional/mandatory key names.
-
{} extends Pick
,>
. If K is an optional field, leave it (OptionalKeys, or remove them if they are RequiredKeys).
-
How to eliminate? Never, of course.
export type RequiredKeys = { [K in keyof T]-? : {} extends Pick<T, K> ? never : K; }[keyof T];
In this case, the optional fields are removed, so the OptionalKeys are retained:
export type OptionalKeys<T> = { [K in keyof T]-? : {} extends Pick<T, K> ? K : never; }[keyof T];Copy the code
The IMmutableKeys field is similar to the MutableKeys field.
interface MutableKeys {
readonlyKeys: never;
notReadonlyKeys: "notReadonlyKeys";
}
Copy the code
Then get the name of the field that is not never.
First, define a tool type IfEqual and compare the two types to each other, even before and after the modification, which is read only and not read only.
type Equal<X, Y, A = X, B = never> = (<T>() => T extends X ? 1 : 2) extends < T >() => T extends Y ? 1:2? A : B;Copy the code
-
Don’t get
() => T extends X? 1:2 interference, understood as the wrapper for comparison, distinguishes between read-only and non-read-only attributes. (
() => T extends X? 1:2) this part, only if the type arguments X are exactly the same, two (
() => T extends X? 1:2) will be congruent, this consistency requires read-only, optional and other modifiers to be consistent.
-
In actual use (in the case of non-read-only, for example), we pass in the interface for X and the interface with the read-only attribute removed -readonly for Y, so that all keys are compared once with the key with the read-only attribute removed. Pass in the field name for A, and in B we need never, so we can leave it out.
Example:
export type MutableKeys<T extends object> = { [P in keyof T]-? : Equal< { [Q in P]: T[P] }, { -readonly [Q in P]: T[P] }, P, never >; }[keyof T];Copy the code
A few easy to get around:
-
The generic Q is not actually used here, just a placeholder for the field of the mapping type.
-
X and Y also have distributed condition types to compare before and after removing readonly fields.
The same goes for:
export type IMmutableKeys<T extends object> = { [P in keyof T]-? : Equal< { [Q in P]: T[P] }, { -readonly [Q in P]: T[P] }, never, P >; }[keyof T];Copy the code
-
Instead of operating on the readonly modifier, this is a judgment statement that switches the condition type.
Pick and Omit based on value type
Before we implement Pick and Omit based on key name, suppose now we need to select and Omit according to value type?
T[K] extends ValueType
export type PickByValueType<T, ValueType> = Pick< T, { [Key in keyof T]-? : T[Key] extends ValueType ? Key : never }[keyof T] >; export type OmitByValueType<T, ValueType> = Pick< T, { [Key in keyof T]-? : T[Key] extends ValueType ? never : Key }[keyof T] >;Copy the code
Condition types take on too much…
Tool Types overview
To summarize the types of tools we wrote above:
-
Partial Readonly(Immutable) is Required, and the corresponding recursive version.
-
Cropped interface: Pick Omit PickByValueType OmitByValueType
-
Infer: ReturnType ParamType PromiseType
-
FunctionKeys OptionalKeys RequiredKeys…
Note that sometimes a single tool type is not sufficient for your needs; you may need multiple tool types to work together, such as using FunctionKeys + Pick to get a field of type function in an interface.
In addition, in fact, some of the above tool types can be implemented with the remapping capability more elegant, do not try?
Due to space constraints (at this point, this article has reached 1.3w words), the type of tools for Typefest will have to be released, but I encourage you to read the source code. Compared with utility-types above, it is more grounded and more interesting to implement.
Some of the new features in TypeScript 4.x
This section is a new addition to the previous release and includes some of the new features introduced in the 4.1 to 4.4 (Beta) releases that are relevant to the content of this article, including template literal types and remapping.
Template literal types
TypeScript 4.1 introduces template literal types, allowing us to construct literal types using the ${} syntax, such as:
type World = 'world';
// "hello world"
type Greeting = `hello ${World}`;
Copy the code
Template literal types also support distributed conditional types, such as:
export type SizeRecord<Size extends string> = `${Size}-Record` // "Small-Record" type SmallSizeRecord = SizeRecord<"Small"> // "Middle-Record" type MiddleSizeRecord = SizeRecord<"Middle"> // "Huge-Record" type HugeSizeRecord = SizeRecord<"Huge"> // "Small-Record" | "Middle-Record" | "Huge-Record" type UnionSizeRecord = SizeRecord<"Small" | "Middle" | "Huge">Copy the code
Interestingly, a template slot (${}) can be passed in union types, and if there are multiple slots in the same template, the union types will be arranged separately.
// "Small-Record" | "Small-Report" | "Middle-Record" | "Middle-Report" | "Huge-Record" | "Huge-Report"
type SizeRecordOrReport = `${"Small" | "Middle" | "Huge"}-${"Record" | "Report"}`;
Copy the code
Four new tool types come with it:
type Uppercase<S extends string> = intrinsic;
type Lowercase<S extends string> = intrinsic;
type Capitalize<S extends string> = intrinsic;
type Uncapitalize<S extends string> = intrinsic;
Copy the code
What they do is literally, and I won’t explain it. For PR, see 40336. Anders Hejlsberg is the chief architect of C# and Delphi, and one of the authors of TS.
Intrinsic means that these tool types are implemented internally by the TS compiler, and it makes sense that we can’t change the value of a literal through type programming, but I think it’s possible that in the future, TS type programming will support calling Lodash methods.
TS implementation code:
function applyStringMapping(symbol: Symbol, str: string) { switch (intrinsicTypeKinds.get(symbol.escapedName as string)) { case IntrinsicTypeKind.Uppercase: return str.toUpperCase(); case IntrinsicTypeKind.Lowercase: return str.toLowerCase(); case IntrinsicTypeKind.Capitalize: return str.charAt(0).toUpperCase() + str.slice(1); case IntrinsicTypeKind.Uncapitalize: return str.charAt(0).toLowerCase() + str.slice(1); } return str; } Copy the code
You might be thinking, well, what happens to a template literal if you want to intercept a part of it? You can’t call the slice method here. In fact, infer is just the idea we mentioned above. By using infer, we can extract part of the literal content, such as:
type CutStr<Str extends string> = Str extends `${infer Part}budu` ? Part : never
// "lin"
type Tmp = CutStr<"linbudu">
Copy the code
Further, with strings like [1,2,3], if we provide slot matching like [${infer Member1}, ${infer Member2}, ${infer Member}], we can achieve the magical effect of extracting array members of the string:
type ExtractMember<Str extends string> = Str extends `[${infer Member1}, ${infer Member2}, ${infer Member3}]` ? [Member1, Member2, Member3] : unknown;
// ["1", "2", "3"]
type Tmp = ExtractMember<"[1, 2, 3]">
Copy the code
Note that the template slots are used and separated. If several of the infer slots are close together, the earlier infer will only acquire a single character and the last infer will acquire all the remaining characters (if any). For example, let’s change the example above to:
type ExtractMember<Str extends string> = Str extends `[${infer Member1}${infer Member2}${infer Member3}]` ? [Member1, Member2, Member3] : unknown;
// ["1", ",", " 2, 3"]
type Tmp = ExtractMember<"[1, 2, 3]">
Copy the code
This property allows us to use multiple adjacent infer + slots and recursively operate on the value obtained by the last infer, as in:
type JoinArrayMember<T extends unknown[], D extends string> = T extends [] ? '' : T extends [any] ? `${T[0]}` : T extends [any, ...infer U] ? `${T[0]}${D}${JoinArrayMember<U, D>}` : string; // "" type Tmp1 = JoinArrayMember<[], '.'>; // "1" type Tmp3 = JoinArrayMember<[1], '.'>; / / "2" type Tmp2 = JoinArrayMember < [1, 2, 3, 4], '. '>;Copy the code
The principle is simple: add the first member of an array each time, do nothing on the last member, and return an empty string on the last match ([]).
Or vice versa? Return 1.2.3.4 to array form?
type SplitArrayMember<S extends string, D extends string> =
string extends S ? string[] :
S extends '' ? [] :
S extends `${infer T}${D}${infer U}` ? [T, ...SplitArrayMember<U, D>] :
[S];
type Tmp11 = SplitArrayMember<'foo', '.'>; // ['foo']
type Tmp12 = SplitArrayMember<'foo.bar.baz', '.'>; // ['foo', 'bar', 'baz']
type Tmp13 = SplitArrayMember<'foo.bar', ''>; // ['f', 'o', 'o', '.', 'b', 'a', 'r']
type Tmp14 = SplitArrayMember<any, '.'>; // stri
Copy the code
Finally, when looking at forms like A.B.C, you should think of Lodash’s GET method, which quickly gets nested properties in the form GET ({},” A.B.C “). But how do you provide type declarations? With the template literal type, you just need to add the infer + condition type.
type PropType<T, Path extends string> =
string extends Path ? unknown :
Path extends keyof T ? T[Path] :
Path extends `${infer K}.${infer R}` ? K extends keyof T ? PropType<T[K], R> : unknown :
unknown;
declare function getPropValue<T, P extends string>(obj: T, path: P): PropType<T, P>;
declare const s: string;
const obj = { a: { b: {c: 42, d: 'hello' }}};
getPropValue(obj, 'a'); // { b: {c: number, d: string } }
getPropValue(obj, 'a.b'); // {c: number, d: string }
getPropValue(obj, 'a.b.d'); // string
getPropValue(obj, 'a.b.x'); // unknown
getPropValue(obj, s); // unknown
Copy the code
Heavy mapping
This ability in TS 4.1 introduction, provides the redirection in mapping type mapping source to a new type of ability, the new type can be a tool of returned results, literal template type, etc., are used to solve when using mapping type, we want to filter/copy of the new interface members, usually to convert the original interface member keys as a new method parameters, Such as:
type Getters<T> = {
[K in keyof T as `get${Capitalize<string & K>}`]: () => T[K]
};
interface Person {
name: string;
age: number;
location: string;
}
type LazyPerson = Getters<Person>;
Copy the code
The result after conversion:
type LazyPerson = {
getName: () => string;
getAge: () => number;
getLocation: () => string;
}
Copy the code
The string here & k because of heavy mapping transformation method (that is, as the latter part of) must be assigned to a string | number | symbol, and k from the keyof, may contain symbol type, This cannot be given to template literals.
If the transformation method returns never, the member is removed, so we can use this method to filter out members.
type RemoveKindField<T> = {
[K in keyof T as Exclude<K, "kind">]: T[K]
};
interface Circle {
kind: "circle";
radius: number;
}
// type KindlessCircle = {
// radius: number;
// }
type KindlessCircle = RemoveKindField<Circle>;
Copy the code
Finally, when used with template literals, because of their permutation properties, if the transformation method of the remap is a union type composed of template literal types, multiple members will be obtained from permutation combinations.
type DoubleProp<T> = { [P in keyof T & string as `${P}1` | `${P}2`]: T[P] }
type Tmp = DoubleProp<{ a: string, b: number }>; // { a1: string, a2: string, b1: number, b2: number }
Copy the code
The end of the
This article is really very long, because it is not recommended to read it whole at one time, it is recommended to select a few paragraphs with a certain length of continuous time, to break it up and knead it to read well. Writing is hard, especially when it’s this long, but if it helps you take TypeScript to the next level, it’s worth it.
If you’ve never focused on the genre aspect of programming before, it may take some time to get used to the change in thinking after reading it. Again, recognize that type programming is also programming. Of course, you can start doing this gradually. For example, you can start today with your current projects, moving from generics to type guards, from index/map types to conditional types, and from utility types to wrapper types to become a TypeScript pro.