项目中常见的 Typescript 语法
# 1. 常用类型
# string 字符串类型
export const str: string = "helloworld";
str.substr(3);
1
2
2
# number 数字类型
let num: number = 100;
num++;
1
2
2
# boolean 布尔类型
const bool: boolean = true;
1
# 数组类型
const numArr: number[] = [1, 2, 3];
numArr.map((num) => ++num);
1
2
2
# 对象类型
type User = {
name: string;
age: number;
isAdmin: boolean;
};
const user: User = {
name: "xiaoming",
age: 18,
isAdmin: false
};
const { name, age, isAdmin } = user;
1
2
3
4
5
6
7
8
9
10
11
2
3
4
5
6
7
8
9
10
11
# 函数类型
type Fn = (n: number) => number;
const fn: Fn = (num) => ++num;
fn(1);
1
2
3
2
3
# 2. React 组件 Props
interface Props {
disabled?: boolean;
style?: React.CSSProperties;
children?: React.ReactNode;
onClick?: () => void;
}
const Button = ({ onClick, disabled, children, style }: Props) => {
return (
<button onClick={onClick} disabled={disabled} style={style}>
{children}
</button>
);
};
export default Button;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 3. 联合类型 Union
// id 可为字符串或数字类型
export function printId(id: string | number) {
console.log(id);
}
printId(101); // OK
printId('202'); // OK
1
2
3
4
5
6
7
2
3
4
5
6
7
# 4. 类型判断
export function printId(id: string | number) {
if (typeof id === 'string') {
console.log(id.toUpperCase());
} else {
console.log(id);
}
}
printId(101); // OK
printId('202'); // OK
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
# 5. 类型断言
export type Position = 'left' | 'right' | 'top' | 'bottom';
const setPos = (pos: Position) => {
//...
};
const handleChange = (value: string) => {
setPos(value as Position);
};
handleChange('left');
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
# 6. 属性名不确定的对象
export type Paths = {
[key: string]: string;
};
// 等同于
// export type Paths = Record<string, string>;
const paths: Paths = {};
paths.home = '/home'; //OK
paths.settings = '/settings'; //OK
paths.somePath = '/somePath'; //OK
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
# 7. 对象的 key 值
export const ErrorMessage = {
0: "success",
7: "Permission denied",
9: "Invalid parameters"
//...
};
export type ErrorCode = keyof typeof ErrorMessage;
export const logErrMsg = (code: ErrorCode) => {
console.log(ErrorMessage[code]);
};
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
# 8. 泛型 generics
type UseState = <T>(v: T) => [T, (v: T) => void];
const useState: UseState = (v) => {
return [
v,
(v) => {
//...
}
];
};
export const Component = () => {
const [num, setNum] = useState(0); // OK
const [str, setStr] = useState(""); // OK
const [list, setList] = useState([1, 2, 3]); // OK
// test
const newNum = num + 1;
setNum(newNum);
const newStr = str.toUpperCase();
setStr(newStr);
const newList = list.slice(1);
setList(newList);
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
# 9. 部分对象 Partial
interface User {
name: string;
age: number;
occupation: string;
}
export const users: User[] = [
{
name: "Max Mustermann",
age: 25,
occupation: "Chimney sweep"
},
{
name: "Wilson",
age: 23,
occupation: "Ball"
}
];
type Criteria = {
[Property in keyof User]?: User[Property];
};
// 等同于
// type Criteria = Partial<User>;
export const filterUsers = (users: User[], criteria: Criteria): User[] =>
users.filter((user) => {
const criteriaKeys = Object.keys(criteria) as (keyof Criteria)[];
return criteriaKeys.every((fieldName) => {
return user[fieldName] === criteria[fieldName];
});
});
const usersOfAge23 = filterUsers(users, {
age: 23
});
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
# 10. 函数中 this 的使用
// 参考 https://www.typescriptlang.org/docs/handbook/2/functions.html#declaring-this-in-a-function
export const debounce = <F extends (...args: any[]) => void>(
fn: F,
delay = 200
) => {
let timeout = 0;
return function (this: any, ...args: any[]) {
timeout && clearTimeout(timeout);
timeout = window.setTimeout(() => {
fn.apply(this, args);
}, delay);
} as F;
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
2
3
4
5
6
7
8
9
10
11
12
13
14
# 11. 类型编程-从基础类型构造新类型
export type CustomObject<K extends string | number, T> = { [key in K]: T };
// 1. 示例
// ObjectOfStringValue 对象的值只能为字符串类型
type ObjectOfStringValue = CustomObject<string, string>;
const objOfStringValue: ObjectOfStringValue = {
h: "hello", // OK
w: "world" // OK
};
// 2. ObjectOfStringValue
// ObjectOfStringValue 对象的值只能为数字类型
type ObjectOfNumberValue = CustomObject<string, number>;
const objOfNumberValue: ObjectOfNumberValue = {
a: 100, // OK
b: 100 // OK
};
const a = objOfNumberValue.a;
// 3. ObjectOfUserValue
type User = {
username: string;
age: number;
};
// ObjectOfUserValue 对象的值只能为User类型
type ObjectOfUserValue = CustomObject<string, User>;
const objOfUserValue: ObjectOfUserValue = {
u1: {
username: "xiaoming",
age: 18
}
};
const { username } = objOfUserValue.u1;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
# 12. 对象类型的继承
export interface Response {
data: any;
status: number;
statusText: string;
}
// 1. 在 Response 类型的基础上添加 config 属性
export interface ResponseWithConfig extends Response {
config: any;
}
const responseWithConfig: ResponseWithConfig = {
data: 100,
status: 0,
statusText: "success",
config: {}
};
// 2. 在 Response 类型的基础上改写 data 属性类型
export interface StringResponse extends Response {
data: string;
}
const stringResponse: StringResponse = {
data: "100",
status: 0,
statusText: "success"
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
# 13. 对象类型的修改
/* extends可以继承对象类型,但不可与原类型冲突,此时可以先使用 Omit 去除需要修改的属性 */
export interface TreeNode {
id: number;
value: number;
children?: TreeNode[];
}
// 1. 去除 TreeNode 的 id 属性同时修改 children 属性的类型
export interface NodeWithoutId extends Omit<TreeNode, "id" | "children"> {
children?: NodeWithoutId[];
}
// OK
const nodeWithoutId: NodeWithoutId = {
value: 1,
children: [
{
value: 2
}
]
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
# 14. 类型编程-条件判断
export declare type Person<T extends "User" | "Admin"> = T extends "User"
? {
username: string;
}
: {
username: string;
role: string;
};
const user: Person<"User"> = { username: "xiaoming" }; // OK
const admin: Person<"Admin"> = { username: "xiaofang", role: "manager" }; // OK
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
# 15. React 组件 Props 范型
import { useState } from "react";
// value 可为字符串或数字
type Value = number | string;
interface Props<T extends Value> {
value?: T;
onChange?: (v: T) => void;
type?: "number" | "text";
}
const Input = <T extends Value>({
value,
onChange,
type = "text"
}: Props<T>) => {
return (
<input
value={value}
onChange={(e) => {
const { value } = e.target;
onChange?.((type === "number" ? parseInt(value, 10) : value) as T);
}}
type={type}
/>
);
};
// test
const Test = () => {
const [num, setNum] = useState(0);
const [str, setStr] = useState("");
return (
<div>
<Input value={num} onChange={(v) => setNum(v)} type="number" />
<Input value={str} onChange={(v) => setStr(v)} />
</div>
);
};
export default Input;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
上次更新: 2024/01/30, 00:35:17
- 01
- linux 在没有 sudo 权限下安装 Ollama 框架12-23
- 02
- Express 与 vue3 使用 sse 实现消息推送(长连接)12-20