Typescript+React入门

初识Typescript

出现背景

Typescript(以下简称TS)实际上就是JavaScript+Type,用数据类型的方式来约束了JS的变量定义

在JS的基础上增加了类型支持

在JS中大多数错误都是因为数据类型造成的,所以TS为了规避这个问题加入了类型限制+编译检查,将问题在代码编译的时候(代码执行前)就可以发现错误

PS:TS是微软开发的,所以作为亲儿子,配合Vscode,TS 可以提前到在编写代码的同时就发现代码中的错误,减少找 Bug、改 Bug 时间

几个中大型框架对TS的支持:

  • Vue 3 源码使用 TS 重写
  • Angular 默认支持 TS
  • React 与 TS 完美配合

TypeScript 已成为大中型前端项目的首先编程语言

快速起步

TS默认是浏览器无法认识的,所以要通过编译器翻译成JS

新建一个文件夹在Vscode,初始化环境

javascript 复制代码
// 安装ts环境
npm i -g typescript
// 查看ts版本(验证是否安装成功)
tsc -v

感受TS编译

新建一个hello.ts 的ts文件

通过命令编译ts文件,生成一个js文件

javascript 复制代码
tsc hello.ts(此时,在同级目录中会出现一个同名的 JS 文件)

执行js代码node .\hello.js

简化编译步骤

每次都这么手动编译太累了,采用自动编译

简化方式:使用 ts-node 包,直接在 Node.js 中执行 TS 代码。

安装命令:npm i -g ts-node(ts-node 包提供了 ts-node 命令)

使用方式:ts-node hello.ts

解释:ts-node 命令在内部偷偷的将 TS -> JS,然后,再运行 JS 代码
报错解决方案

当然,真正工程里初始化是完美自动化的,具体实现可以自行百度

TS常用类型

前面我们知道了:

TypeScript 是 JS 的超集,TS 提供了 JS 的所有功能,并且额外的增加了类型系统(JS中,如果number在运行过程中突然变成了boolean是很常见的,所以也非常容易出问题)

类型注解

在哪用类型注解

示例:

如图所示:代码中的 : number 就是类型注解。

目的就是为了给age加上类型的约束,一旦变为其他的类型则报错

就会变成这样,so,约定了什么类型,就只能给变量赋值该类型的值,否则,就会报错。

常用类型注解有哪些

可以将 TS 中的常用基础类型细分为两类:1 JS 已有类型 2 TS 新增类型

  1. JS 已有类型
    原始类型:number/string/boolean/null/undefined/symbol。
    对象类型:object(包括,数组、对象、函数等对象)。
  2. TS 新增类型
    联合类型、自定义类型(类型别名)、接口、元组、字面量类型、枚举、void、any 等

原始类型

原始类型:number/string/boolean/null/undefined/symbol。

这些按照上面的写法来做即可

数组类型

对象类型:object(包括,数组、对象、函数等对象)

数组的两种写法:

javascript 复制代码
//数组的定义方式这两种均可,更推荐第一个
let numbers0: number[] = [1, 2, 3];
let numbers1: Array<number> = [1, 2, 3];
//换了个数据类型
let numbers2: string[] = ["1", "2", "3"];
let numbers3: Array<string> = ["1", "2", "3"];

需求:数组里面我既要存number,又要存string,怎么办

| 来隔开数据类型

javascript 复制代码
let numbers4: (number | string)[] = [1, "2", 3];
let numbers5: Array<number | string> = [1, "2", 3];

解释:| (竖线)在 TS 中叫做联合类型(由两个或多个其他类型组成的类型,表示可以是这些类型中的任意一种)。

注意:这是 TS 中联合类型的语法,只有一根竖线,不要与 JS 中的或(||)混淆了

类型别名

类型别名(自定义类型):为任意类型起别名。

使用场景:当同一类型(复杂)被多次使用时,可以通过类型别名,简化该类型的使用(相当于抽一个共通的类型出来)

语法:type 自定义类型名=(类型1|类型2|.....)[]注意最后有个数组符号

例子:

javascript 复制代码
type userArray = (number | string)[];
let numbers6: userArray = [1, "2", 3];

解释:

  1. 使用 type 关键字来创建类型别名。
  2. 类型别名(比如,此处的 CustomArray),可以是任意合法的变量名称。
  3. 创建类型别名后,直接使用该类型别名作为变量的类型注解即可。

这里补一个知识,用type类型是可以定义方法头部的(类似接口)
type 函数名 = (param1: number, param2: string) => ReturnType;

javascript 复制代码
//先用type来定义方法的参数列表,箭头函数后面是方法的返回值类型
//type 函数名 = (param1: number, param2: string) => ReturnType;
type F1 = (a: number) => number
// 定义f1,类型用F1约束好,利用箭头函数实现方法体
// 注意,这里只是按照type的约束把方法定义出来,调用的时候需要实例化或者去单独调用
let f1: F1 = (a: number):number => {
    // your code ...
    return a + 1;
}
//方法调用
let result = f1(111)
console.log(result)

函数类型(类似方法定义)

函数的类型实际上指的是:函数参数和返回值的类型。

为函数指定类型的两种方式:1 单独指定参数、返回值的类型 2 同时指定参数、返回值的类型。

单独指定参数、返回值的类型:

语法:

javascript 复制代码
function 函数名(参数1: 类型, 参数2: 类型): 返回值类型{
	//方法体
    return 返回值(如果注明了返回值类型的话就代表有返回值);
}
javascript 复制代码
//这两种定义方法的方法结果是一样的,区别在于定义的方式不同
//function是原始的js玩法,const是用的箭头函数,调用的时候也完全一样
function add(param1: number, param2: number): number {
    return param1 + param2;
}

const add1 = (param1: number, param2: number): number => {
    return param1 + param2;
}

add(1, 2)
add1(1, 2)

同时指定参数、返回值的类型:

同时指定参数、返回值的类型

javascript 复制代码
const add2: (param1: number, param2: number) => number = (param1, param2) => {
    return param1 + param2;
}

解释:当函数作为表达式时,可以通过类似箭头函数形式的语法来为函数添加类型。

注意:这种形式只适用于函数表达式

无返回值类型

如果函数没有返回值,那么,函数返回值类型为:void

javascript 复制代码
function f1(name: string): void {
    console.log("userName:", name)
}

可选参数

使用函数实现某个功能时,参数可以传也可以不传。这种情况下,在给函数参数指定类型时,就用到可选参数了。

比如,数组的 slice 方法,可以 slice() 也可以 slice(1) 还可以 slice(1, 3)

自定义一个测试类

javascript 复制代码
function f2(name1?: string, name2?: string): void {
    console.log("name1:", name1, "name2:", name2);
}
//调用时传不传参数都可以
f2("aa", "bb");
f2("aa");

可选参数:在可传可不传的参数名称后面添加 ?(问号)

注意:可选参数只能出现在参数列表的最后,也就是说可选参数后面不能再出现必选参数。也就是function f2(name1: string, name2?: string):void{ ... }

对象类型

JS 中的对象是由属性和方法构成的,而 TS 中对象的类型就是在描述对象的结构(有什么类型的属性和方法)。

对象类型的写法:

javascript 复制代码
let obj1: {name: string,sayHi(): void} = {
    name: "张三",
    sayHi(): void { console.log("Hi~") },
}

解释:

  1. 直接使用 {} 来描述对象结构。属性采用属性名: 类型的形式;方法采用方法名(): 返回值类型的形式。
  2. 如果方法有参数,就在方法名后面的小括号中指定参数类型(比如:greet(name: string): void)。
  3. 在一行代码中指定对象的多个属性类型时,使用 ;(分号)来分隔。
  • 如果一行代码只指定一个属性类型(通过换行来分隔多个属性类型),可以去掉 ;(分号)。

  • 方法的类型也可以使用箭头函数形式(比如:{ sayHi: () => void })。

再扩展几个方法

javascript 复制代码
let obj1: {
    name: string,
    age: number,
    sayHi(): void,
    sayName(name: string): void
} = {
    name: "张三",
    age: 18,
    sayHi(): void { console.log("Hi~") },
    sayName(name: string): void { console.log("name:", name) }
}

对象类型(参数可选情况)

对象的属性或方法,也可以是可选的,此时就用到可选属性了。

比如,我们在使用 axios({ ... }) 时,如果发送 GET 请求,method 属性就可以省略。

javascript 复制代码
function myAxios(config: { url: string, method?: string }) {
    //灵活应用,如果发送 GET 请求,method 属性就可以省略
    //打印的时候如果没传对应的参数就不打印了
    console.log(config)
}

可选属性的语法与函数可选参数的语法一致,都使用 ?(问号)来表示

接口

当对象类型被多次使用时,一般会使用接口(interface)来描述对象的类型,来达到复用的目的。

类似继承,方法不实现是不可以的,属性,方法都要具体定义。

修改后,对变量,方法做具体实现

javascript 复制代码
// 定义接口,定义好属性
interface IPerson {
    name: string;
    age: number;
    sayHi(): void;
}
//使用接口类型约束后,需要具体实现
const obj: IPerson = {
    name: "张三",
    age: 18,
    sayHi(){
        console.log("Hi~")
    }
}

解释:

  1. 使用 interface 关键字来声明接口。
  2. 接口名称(比如,此处的 IPerson),可以是任意合法的变量名称。
  3. 声明接口后,直接使用接口名称作为变量的类型。
  4. 因为每一行只有一个属性类型,因此,属性类型后没有 ;(分号)。

interface和type的区别

接口更灵活

相同点:都可以给对象指定类型。

  • 不同点:接口,只能为对象指定类型。
  • 类型别名,不仅可以为对象指定类型,实际上可以为任意类型指定别名。

接口复用

如果两个接口之间有相同的属性或方法,可以将公共的属性或方法抽离出来,通过继承来实现复用。

原有状态:

javascript 复制代码
interface Point2D {
    x: number;
    y: number;
}

interface Point3D {
    x: number;
    y: number;
    z: number;
}

比如,这两个接口都有 x、y 两个属性,重复写两次,可以,但很繁琐。

所以就引出了接口复用。

直接用extends来复用2D里面的内容

javascript 复制代码
interface Point3D extends Point2D {
    z: number;
}
//集成后对值进行实现
const a: Point3D = {
    x: 1,
    y: 2,
    z: 3,
}

解释:

  1. 使用 extends(继承)关键字实现了接口 Point3D 继承 Point2D。
  2. 继承后,Point3D 就有了 Point2D 的所有属性和方法(此时,Point3D 同时有 x、y、z 三个属性)。

元组

场景:在地图中,使用经纬度坐标来标记位置信息。

可以使用数组来记录坐标,那么,该数组中只有两个元素,并且这两个元素都是数值类型。

javascript 复制代码
//这样标记不严谨,没有具体约束有多少个元素
let position: number[] = [1, 2]
//使用元组进行约束,固定两个number元素的数组(当然,其他元素也完全可以,随便搞)
let position1: [number, number] = [1, 2]

元组类型是另一种类型的数组,它确切地知道包含多少个元素,以及特定索引对应的类型

解释:

  1. 元组类型可以确切地标记出有多少个元素,以及每个元素的类型。
  2. 该示例中,元素有两个元素,每个元素的类型都是 number。

类型推断

TS 中,某些没有明确指出类型的地方,TS 的类型推论机制会帮助提供类型。

换句话说:由于类型推论的存在,这些地方,类型注解可以省略不写! 发生类型推论的 2 种常见场景:1 声明变量并初始化时 2 决定函数返回值时。

鼠标放在上面自动推断

类型断言

有时候你会比 TS 更加明确一个值的类型,此时,可以使用类型断言来指定更具体的类型。(我断言这是一个xxx的类型)

用途一般都是获取某某标签,然后通过断言来获取标签属性

注意:getElementById 方法返回值的类型是 HTMLElement,该类型只包含所有标签公共的属性或方法,不包含 a

标签特有的 href 等属性。

因此,这个类型太宽泛(不具体),无法操作 href 等 a 标签特有的属性或方法。

解决方式:这种情况下就需要使用类型断言指定更加具体的类型。

使用类型断言:

明确的指明元素的类型

解释:

  1. 使用 as 关键字实现类型断言。
  2. 关键字 as 后面的类型是一个更加具体的类型(HTMLAnchorElement 是 HTMLElement 的子类型)。
  3. 通过类型断言,aLink 的类型变得更加具体,这样就可以访问 a 标签特有的属性或方法了。

通过控制台打印

字面量类型

首先看下面两个变量类型

并不是都是string类型的

javascript 复制代码
let str1='Hello TS'
const str2='Hello TS!'

通过 TS 类型推论机制,可以得到答案:

  1. 变量 str1 的类型为:string。
  2. 变量 str2 的类型为:'Hello TS'。

解释:

  1. str1 是一个变量(let),它的值可以是任意字符串,所以类型为:string。

  2. str2 是一个常量(const),它的值不能变化,只能是 'Hello TS',所以,它的类型为:'Hello TS'。这个变量锁死了就只能是**'Hello TS'**,const代表常量,不可变更

注意:此处的 'Hello TS',就是一个字面量类型。也就是说某个特定的字符串也可以作为 TS 中的类型。 除字符串外,任意的 JS 字面量(比如,对象、数字等)都可以作为类型使用。

使用模式:字面量类型配合联合类型一起使用。

使用场景:用来表示一组明确的可选值列表。

比如,在贪吃蛇游戏中,游戏的方向的可选值只能是上、下、左、右中的任意一个(这里用到了枚举)。

javascript 复制代码
//这里用了枚举,所以changeDirection的参数只能传'up' | 'down' | 'left' | 'right'中的一个
function changeDirection(direction: 'up' | 'down' | 'left' | 'right') {
    console.log(direction)
}
//只能传'up' | 'down' | 'left' | 'right'中的一个
changeDirection('up');

严格的类型约束

枚举

基础使用

上面字面量的场景,当字面量很多的情况下,就会显得很冗余

这里就换成枚举

枚举的功能类似于字面量类型+联合类型组合的功能,也可以表示一组明确的可选值。

枚举:定义一组命名常量。它描述一个值,该值可以是这些命名常量中的一个。

  1. 使用 enum 关键字定义枚举。
  2. 约定枚举名称、枚举中的值以大写字母开头。
  3. 枚举中的多个值之间通过 ,(逗号)分隔。
  4. 定义好枚举后,直接使用枚举名称作为类型注解。

调用过程中,因为标记了只能传入枚举类型,so,传入的时候只能枚举.值

数字枚举

为什么叫数字枚举?

不难发现,枚举本身是没有对值进行定义的

而枚举的默认类型是number,并且第一个元素默认的值是0

第n个元素的默认值是n-1

这是数字枚举的默认情况,当然我们可以像Java里的枚举一样,进行默认值赋值处理。

如果有的枚举没有赋值,那么他就会默认继续自增下去

字符串枚举

看字面量就可以知道,字符串枚举内部的类型都是字符串(需要默认就给赋值,不赋值就会被默认为数字枚举)

如果有其中一个没有赋值,就会报错,因为:
字符串枚举没有自增长行为,因此,字符串枚举的每个成员必须有初始值。

枚举编译后的样子

any类型

首先说明:any类型不推荐在TS使用,因为any会把TS变成anyScript,

当一个变量被标记为any的时候,那么将不再会对其有任何提示

当用any标记之后,退化成JS了属于是😂

编译才会发现错误,违背了TS的早发现初衷

尽可能的避免使用 any 类型,除非临时使用 any 来"避免"书写很长、很复杂的类型!

其他隐式具有 any 类型的情况:

1 声明变量不提供类型也不提供默认值

2 函数参数不加类型。

注意:因为不推荐使用 any,所以,这两种情况下都应该提供类型!

typeof


实际上,TS 也提供了 typeof 操作符:可以在类型上下文中引用变量或属性的类型(类型查询)。

使用场景:根据已有变量的值,获取该值的类型,来简化类型书写

未简化前状态

既然上面的let p和方法里面的point对象是一个类型的,那么就可以用typeof简化

这里需要指明关键字,直接标记是不可以的

优化后,使用typeof关键字

解释:

  1. 使用 typeof 操作符来获取变量 p 的类型,结果与第一种(对象字面量形式的类型)相同。
  2. typeof 出现在类型注解的位置(参数名称的冒号后面)所处的环境就在类型上下文(区别于 JS 代码)。
  3. 注意:typeof 只能用来查询变量或属性的类型,无法查询其他形式的类型(比如,函数调用的类型)

TS高级类型

class类

这个和Java类差不多

TypeScript 全面支持 ES2015 中引入的 class 关键字,并为其添加了类型注解和其他语法(比如,可见性修饰符等)

  1. 根据 TS 中的类型推论,可以知道 Person 类的实例对象 p 的类型是 Person。
  2. TS 中的 class,不仅提供了 class 的语法功能,也作为一种类型存在。

class类初始化


class构造函数

解释:

  1. 成员初始化(比如,age: number)后,才可以通过 this.age 来访问实例成员。
  2. 需要为构造函数指定类型注解,否则会被隐式推断为 any;构造函数不需要返回值类型。

此时再新建对象就可以直接指定值

class类的方法

在class类中提供一些方法,创建完成之后通过创建对象调用

解释:方法的类型注解(参数和返回值)与函数用法相同,正常传参调用即可。(当然,那个void可以不写,直接靠return自动类型推断也完全OK)

class类继承(extends)

这个是js就自带的,通过继承类,就可以调用父类的方法,通过继承获取父类的所有属性以及方法

解释:

  1. 通过 extends 关键字实现继承。
  2. 子类 Dog 继承父类 Animal,则 Dog 的实例对象 dog 就同时具有了父类 Animal 和 子类 Dog 的所有属性和方法。
  3. Dog类新建对象调用方法

class类实现(implements)

这个是ts提供的,通过interface定义类,用implements来实现,在子类中对方法进行具体实现

解释:

  1. 通过 implements 关键字让 class 实现接口。
  2. sing类实现接口 Singable 意味着,sing 类中必须提供 Singable 接口中指定的所有方法和属性

class类访问权限控制

相比于Java的四种控制权限,TS只有3种,少了Java的default

类成员可见性:可以使用 TS 来控制 class 的方法或属性对于 class 外的代码是否可见。 可见性修饰符包括:1 public(公有的) 2 protected(受保护的) 3 private(私有的)。

public(公有的)

public:表示公有的、公开的,公有成员可以被任何地方访问,默认可见性

在哪都能访问到就不具体演示了

解释:

  1. 在类属性或方法前面添加 public 关键字,来修饰该属性或方法是共有的。
  2. 因为 public 是默认可见性,所以,可以直接省略。

protected(受保护的)

protected:表示受保护的,仅对其声明所在类和子类中(非实例对象)可见。

解释:

  1. 在类属性或方法前面添加 protected 关键字,来修饰该属性或方法是受保护的。
  2. 子类的方法内部 可以通过 this 来访问父类中受保护的成员,但是,对实例不可见!

private(私有的)

private:表示私有的,只在当前类中(方法内)可见,对实例对象以及子类也是不可见的。

解释:

  1. 在类属性或方法前面添加 private 关键字,来修饰该属性或方法是私有的。
  2. 私有的属性或方法只在当前类中的方法可见,对子类和实例对象也都是不可见的!

readonly(只读修饰符)

除了可见性修饰符之外,还有一个常见修饰符就是:readonly(只读修饰符)。

readonly:表示只读,用来防止在构造函数之外对属性进行赋值。

解释:

  1. 使用 readonly 关键字修饰该属性是只读的,注意只能修饰属性不能修饰方法
  2. 注意:属性 age 后面的类型注解(比如,此处的 number)如果不加,则 age 的类型为 18 (字面量类型)。
  3. 接口或者 {} 表示的对象类型,也可以使用 readonly来修饰变量

类型兼容性

先说个梗,如果一个东西,走路像鸭子,长得像鸭子,吃饭像鸭子,那他就是个鸭子

两种类型系统:1 Structural Type System(结构化类型系统) 2 Nominal Type System(标明类型系统)。

为了帮助理解,这种其实是类似于多态,很大程度上利用了class、方法的兼容

Structural Type System(结构化类型系统)

TS 采用的是结构化类型系统,也叫做 duck typing(鸭子类型),类型检查关注的是值所具有的形状。 也就是说,在结构类型系统中,如果两个对象具有相同的形状,则认为它们属于同一类型(它看起来就是鸭子)。

这两个可以被认为是同一类型

那么point和point2的两个对象的结构是相同的,因此认为是"同一类型",在new对象的时候就可以这样

javascript 复制代码
//这里之所以可以这么创建,是因为point和point2的两个对象被认为是同一类型
//这里point代表p的类型,point2()代表创建point2类型的实例
const p: point = new point2()

解释:

  1. Point 和 Point2D 是两个名称不同的类。
  2. 变量 p 的类型被显示标注为 Point 类型,但是,它的值却是 Point2D 的实例,并且没有类型错误。
  3. 因为 TS 是结构化类型系统,只检查 Point 和 Point2D 的结构是否相同,结构相同就认为是同一类型(相同,都具有 x 和 y 两个属性,属性类型也相同)。
  4. 但是,如果在 Nominal Type System 中(比如,C#、Java 等),它们是不同的类,类型无法兼容。

结构化类型系统向下兼容

注意:在结构化类型系统中,如果两个对象具有相同的形状,则认为它们属于同一类型,这种说法并不准确。 更准确的说法:对于对象类型来说,y 的成员至少与 x 相同,则 x 兼容 y(成员多的可以赋值给少的)。

反过来的话肯定不兼容

解释:

  1. Point3D 的成员至少与 Point 相同,则 Point 兼容 Point3D。
  2. 所以,成员多的 Point3D 可以赋值给成员少的 Point(只要能全部满足少的一方,多加多少都可以)。

接口兼容性

除了 class 之外,TS 中的其他类型也存在相互兼容的情况,包括:1 接口兼容性 2 函数兼容性 等。

接口之间的兼容性,类似于 class。并且,class 和 interface 之间也可以兼容。

interface之间 可以直接使用类型兼容,写法和class一样

interface与class之间也可以兼容

函数兼容性

函数之间兼容性比较复杂,需要考虑:1 参数个数 2 参数类型 3 返回值类型


  • 参数个数,参数多的兼容参数少的(或者说,参数少的可以赋值给多的,和对象刚好相反)。
javascript 复制代码
//用type来定义函数
type 函数名 = (param1: number, param2: string) => ReturnType;
javascript 复制代码
//先用type来定义方法的参数列表,箭头函数后面是方法的返回值类型
//type 函数名 = (param1: number, param2: string) => ReturnType;
type F1 = (a: number) => void
// 定义f1,类型用F1约束好,利用箭头函数实现方法体
// 注意,这里只是定义,调用需要实例化或者单独调用方法
let f1: F1 = (a: number): void => {
    // your code ...
}
// 定义F2的方法体,比F1的要多一个参数
type F2 = (a: number, b: number) => void
//将F1的方法比F2的方法要少一个参数,所以可以兼容
let f2: F2 = f1
//方法调用
let result = f2(111)
console.log(result)

再比如说数组的forEach方法,可以选择传一个参数,也可以一个不传

解释:

  • 参数少的可以赋值给参数多的,所以,f1 可以赋值给 f2。
  • 数组 forEach 方法的第一个参数是回调函数,该示例中类型为:(value: string, index: number, array: string[]) => void。
  • 在 JS 中省略用不到的函数参数实际上是很常见的,这样的使用方式,促成了 TS 中函数类型之间的兼容性。
  • 并且因为回调函数是有类型的,所以,TS 会自动推导出参数 item、index、array 的类型。

  • 参数类型,相同位置的参数类型要相同(原始类型)或兼容(对象类型)

    解释:函数类型 F2 兼容函数类型 F1,因为 F1 和 F2 的第一个参数类型相同。

来个复杂点的:

先用interface定义参数类型

在type定义参数列表的时候用上interface做定义

先定义了f2,再去赋给f3,但是f3反过来赋给f2不可以

因为参数列表无法小范围 的去兼容 大范围

解释:

  1. 注意,此处与前面讲到的接口兼容性冲突。
  2. 技巧:将对象拆开,把每个属性看做一个个参数,则,参数少的(f2)可以赋值给参数多的(f3)。

  • 返回值类型,只关注返回值类型本身即可:

    解释:
  1. 如果返回值类型是原始类型,此时两个类型要相同,比如,左侧类型 F5 和 F6。
  2. 如果返回值类型是对象类型,此时成员多的可以赋值给成员少的,比如,右侧类型 F7 和 F8。

交叉类型

交叉类型(&),有点类似于接口继承(extends),用于多个类型组合为一个类型

javascript 复制代码
//定义两个接口类型
interface Person {
  name: string;
}
interface Man {
  age: string;
}
//用type来交叉两个类型
type PersonAndMan = Person & Man;
const p: PersonAndMan = {
  //获取到Person的name属性
  name: "zhangsan",
  //获取到Man的age属性
  age: "lisi",
};

解释:当使用了交叉类型之后,type PersonAndMan就拥有了PersonMan的两个类型


交叉类型(&)与继承(extends)的对比

  • 相同点:都可以实现对对象类型的组合
  • 不同点:两种方式实现类型组合时,对于同名属性之间出现冲突(如果同名同属性,则视为同一个,如果同名不同属性,则会冲突),处理类型冲突的方式不同

泛型

初识泛型

泛型时可以保证类型安全的前提下,让函数等与多种类型一起工作,从而实现灵活复用,常用于函数、接口、class中

需求:创建一个id函数,传入什么数据就返回数据本身(参数和返回值类型一样)

比如这个函数,只能传递number类型并且返回number类型,无法用于其他类型

javascript 复制代码
function id(param: number): number{
  return param;
}

要是稍微改造一下,把类型换成any。这样确实可以接受其他类型,但代价是失去了TS的类型保护,不安全

javascript 复制代码
function id(param: any): any {
  return param;
}

此时的解决方案就是泛型

泛型再保证类型安全(不丢失类型信息)的同时,可以让函数可以传入多种不同的类型,实现灵活复用。

创建一个泛型函数

实操一下

javascript 复制代码
//将函数得参数类型定义为Type
function test<Type>(value: Type): Type {
  return value;
}
//同样的函数传入不同的类型,不会报错,完成同一个函数复用的目的
const m1: number = test(20);
const m2: string = test("aa");
const m3 = test(20);
const m4 = test("aa");

解释:

1.再调用泛型函数时,可以省略<类型>来简化泛型函数的调用

2.此时TS的内部会采用类型参数推断机制,来根据传入的实参自动推断出类型变量Type的类习惯

3.比如此时传入实参20,TS会自动推断出传入的参数时number,此时Type就会自动变成number类型

推荐:使用这种简化的方式调用泛型函数,使代码更短更易于阅读

说明:当编译器无法推断类型或者推断类型不够准确的时候,就需要显式的传入参数类型参数

泛型约束

问题描述

默认情况下,由于Type类型可以代表的类型太多,导致无法访问任何属性

泛型收缩的方式

添加泛型约束来收缩类型,主要有两种方式:1 指定更加具体的类型 2 添加约束

  • 1.指定更加具体的类型
javascript 复制代码
function test2<Type>(value: Type[]): Type[] {
  console.log(value.length);
  return value;
}

比如,将类型修改为Type[](Type类型的数组),因为只要是数组就一定存在length属性,因此就可以访问了。

  • 2.添加约束
    创建 描述约束的接口ILength,该接口要求提供length属性
    通过继承extends关键字来使用该接口,为泛型(类型变量)添加约束
    该约束表示,传入的类型中必须有length属性
    注意,这里传入的实参(比如数组,对象),只要有length属性即可,并且也符合前面讲到的接口类型兼容性
javascript 复制代码
//指定一个number类型的Type
interface ILength {
  length: number;
}
//Type利用extends来约束类型
function test3<Type extends ILength>(param: Type): Type {
  console.log(param.length);
  return param;
}

多个类型变量:

类型变量间约束

泛型接口

javascript 复制代码
//定义接口类型为Type,内部的类型都可以根据Type来变化
interface IdFunc<Type> {
  id: (value: Type) => Type;
  ids: () => Type[];
}
//新建变量实现接口,接口传入number
let obj: IdFunc<number> = {
  id(value) {
    return value;
  },
  ids() {
    return [1, 3, 5];
  },
};
//调用方法...
obj.id(1);

解释:

1.在interface后面添加<类型变量>,那么该接口就变成了泛型接口

2.interface的类型变量,对接口中所有其他成员都可见,也就是接口中所有成员都可以用类型变量

3.使用泛型接口的时候,需要显式的标注具体类型,比如:interface IdFunc<Type> {...}

4.通过变量实现接口后let obj: IdFunc<number> = {},obj的变量和方法都会受到传入number变量的影响。

比如 id: (value: Type) => Type;在obj的实现中就会变成 id: (value: number) => number;

比如 ids: () => number[];在obj的实现中就会变成 ids: () => number[];

实际上,JS中的数组在TS中,就是一个泛型接口,根据传入的不同类型变化为不同的数组类型

每次foreach的方法参数类型也会不一样

泛型类

class也可以配合泛型来使用

比如React中的class组件的基类Component就是泛型类,不同的组件有不同的props和state

创建一个泛型类

在创建类型的时候明确指定创建对象类型

解释:

1.类似于泛型接口,在class名称后面添加 <类型变量>,这个类就变成了泛型类

2.此处的add方法,采用的是箭头函数形式的类型书写方法

泛型工具类

泛型工具类型:TS 内置了一些常用的工具类型,来简化 TS 中的一些常见操作。 说明:它们都是基于泛型实现的(泛型适用于多种类型,更加通用),并且是内置的,可以直接在代码中使用。

这些工具类型有很多,主要学习以下几个:

javascript 复制代码
Partial<Type>
Readonly<Type>
Pick<Type, Keys>
Record<Keys, Type>

Partial< Type>可选

泛型工具类型Partial用来构造(创建)一个类型,将 Type 的所有属性设置为可选

构造出来的新类型 PartialProps 结构和 Props 相同,但所有属性都变为可选的

javascript 复制代码
interface A {
    id: string
    children: number[]
}
//定义Props的Type,结构与interface A一致
//但是所有属性都变为可选的
type Props = Partial<A>
//定义obj对象,结构可以与A一样
let obj: Props = {
    id: "123",
    children: [1, 2, 3]
}

ReadOnly< Type>只读

泛型工具类型 - Readonly 用来构造一个类型,将 Type 的所有属性都设置为 readonly(只读)。

构造出来的新类型 ReadonlyProps 结构和 A 相同

Pick<Type, Keys>可选

泛型工具类型 Pick<Type, Keys> 从 Type 中选择一组属性来构造新类型

javascript 复制代码
interface A {
    id: string
    age: string
    children: number[]
}
//定义一个Pick的类,选择A接口中的两个属性
type ReadonlyType = Pick<A, 'id' | 'age'>
//到这里为止,实际上只能使用A接口的id与age属性
let obj: ReadonlyType = {
    id: "111",
    age: "18"
} 

如果强行加入只会报错

  1. Pick 工具类型有两个类型变量:1 表示选择谁的属性 2 表示选择哪几个属性。
  2. 其中第二个类型变量,如果只选择一个则只传入该属性名即可。
  3. 第二个类型变量传入的属性只能是第一个类型变量中存在的属性。
  4. 构造出来的新类型 PickProps,只有 id 和 title 两个属性类型。

Record<Keys,Type>构造

泛型工具类型Record<Keys,Type> 构造一个对象类型,属性键为 Keys,属性类型为 Type。

Record 工具类型有两个类型变量:1 表示对象有哪些属性 2 表示对象属性的类型。

javascript 复制代码
//用Record来记录类型
//规定RecordObj有'a' | 'b' | 'c'三个属性,均为string数组
type RecordObj = Record<'a' | 'b' | 'c', string[]>
let obj: RecordObj = {
    a: ['1'],
    b: ['2'],
    c: ['3']
}
//规定RecordObj2有'a' | 'b' | 'c'三个属性,均为string类型
type RecordObj2 = Record<'a' | 'b' | 'c', string>
let obj2: RecordObj2 = {
    a: '1',
    b: '2',
    c: '3'
}

索引签名类型

绝大多数情况下,我们都可以在使用对象前就确定对象的结构,并为对象添加准确的类型。

使用场景:当无法确定对象中有哪些属性(或者说对象中可以出现任意多个属性),此时,就用到索引签名类型了。

解释:

  1. 使用 [key: string] 来约束该接口中允许出现的属性名称。表示只要是 string 类型的属性名称,都可以出现在对象中。number代表属性只能赋number类型
  2. 这样,对象 obj 中就可以出现任意多个属性(比如,a、b 等)。
  3. key 只是一个占位符,可以换成任意合法的变量名称。
  4. 隐藏的前置知识:JS 中对象({})的键是 string 类型的。

数组索引签名

在 JS 中数组是一类特殊的对象,特殊在数组的键(索引)是数值类型。

并且,数组也可以出现任意多个元素。所以,在数组对应的泛型接口中,也用到了索引签名类型。

解释:

  1. MyArray 接口模拟原生的数组接口,并使用 [n: number] 来作为索引签名类型。
  2. 该索引签名类型表示:只要是 number 类型的键(索引)都可以出现在数组中,或者说数组中可以有任意多个元素。
  3. 同时也符合数组索引是 number 类型这一前提。

映射类型

映射类型:基于旧类型创建新类型(对象类型),减少重复、提升开发效率。(简单说就是把属性名都给copy过来,具体类型重新定义)

解释:

  1. 映射类型是基于索引签名类型的,所以,该语法类似于索引签名类型,也使用了 []。
  2. Key in PropKeys 表示 Key 可以是 PropKeys 联合类型中的任意一个,类似于 forin(let k in obj)。
  3. 使用映射类型创建的新对象类型 Type2 和类型 Type1 结构完全相同。
  4. 注意:映射类型只能在类型别名中使用,不能在接口中使用。

其他的一些情况:



TypeScript 类型声明文件

今天几乎所有的 JavaScript 应用都会引入许多第三方库来完成任务需求。

这些第三方库不管是否是用 TS 编写的,最终都要编译成 JS 代码,才能发布给开发者使用。

我们知道是 TS 提供了类型,才有了代码提示和类型保护等机制。

但在项目开发中使用第三方库时,你会发现它们几乎都有相应的 TS 类型,这些类型是怎么来的呢?类型声明文件

类型声明文件:用来为已存在的 JS 库提供类型信息。 这样在 TS 项目中使用这些库时,就像用 TS 一样,都会有代码提示、类型保护等机制了。

TS中的两种文件类型

TS 中有两种文件类型:1 .ts 文件 2 .d.ts 文件
.ts 文件:

  1. 既包含类型信息又可执行代码。
  2. 可以被编译为 .js 文件,然后,执行代码。
  3. 用途:编写程序代码的地方。

.d.ts 文件:

  1. 只包含类型信息的类型声明文件。
  2. 不会生成 .js 文件,仅用于提供类型信息。
  3. 用途:为 JS 提供类型信息。

总结:.ts 是 implementation(代码实现文件);.d.ts 是 declaration(类型声明文件)。

如果要为 JS 库提供类型信息,要使用 .d.ts 文件

使用已有的类型声明文件

  • 1 内置类型声明文件

  • 2 第三方库的类型声明文件。

库自带类型声明文件

由 DefinitelyTyped 提供

自己创建声明文件并使用

项目内共享类型

如果多个 .ts 文件中都用到同一个类型,此时可以创建 .d.ts 文件提供该类型,实现类型共享。

操作步骤:

  1. 创建 utils.d.ts 类型声明文件。
  2. 创建需要共享的类型,并使用 export 导出(TS 中的类型也可以使用 import/export 实现模块化功能)。
  3. 在需要使用共享类型的 .ts 文件中,通过 import 导入即可(.d.ts 后缀导入时,直接省略)。

为已有 JS 文件提供类型声明

相当于加载js文件时自动转换为.d.ts文件

React中使用TS

使用 CRA 创建支持 TS 的项目

看起来CRA是什么很屌的工具是吧,实际不是

说白了就是个一件创建React工程的

React 脚手架工具 create-react-app(简称:CRA)默认支持 TypeScript。

创建支持 TS 的项目命令:create-react-app my-react-app 项目名称my-react-app

当看到以下提示时,表示支持 TS 的项目创建成功:

相比于非TS的项目的区别

相对于非 TS 项目,目录结构主要由以下三个变化:

  1. 项目根目录中增加了 tsconfig.json 配置文件:指定 TS 的编译选项(比如,编译时是否移除注释)。
  2. React 组件的文件扩展名变为:*.tsx。
  3. src 目录中增加了 react-app-env.d.ts:React 项目默认的类型声明文件。

    解读一下react-app-env.d.ts:React 项目默认的类型声明文件。
    三斜线指令:指定依赖的其他类型声明文件,types 表示依赖的类型声明文件包的名称

    解释:告诉 TS 帮我加载 react-scripts 这个包提供的类型声明。
    react-scripts 的类型声明文件包含了两部分类型:
  4. react、react-dom、node 的类型
  5. 图片、样式等模块的类型,以允许在代码中导入图片、SVG 等文件。
    TS 会自动加载该 .d.ts 文件,以提供类型声明(通过修改 tsconfig.json 中的 include 配置来验证)

TS配置文件tsconfig.json

tsconfig.json 指定:项目文件和项目编译所需的配置项。 注意:TS 的配置项非常多(100+),以 CRA 项目中的配置为例来学习,其他的配置项用到时查文档即可。

  1. tsconfig.json 文件所在目录为项目根目录(与 package.json 同级)。
  2. tsconfig.json 可以自动生成,命令:tsc --init。一般需要手动生成

可以看到很多的配置被注掉,选择需要的解开即可

除了在 tsconfig.json 文件中使用编译配置外,还可以通过命令行来使用。

使用演示:tsc hello.ts --target es6。

这个的意思是 编译hello.ts文件,以es6的版本编译

注意:

  1. tsc 后带有输入文件时(比如,tsc hello.ts),将忽略 tsconfig.json 文件。
  2. tsc 后不带输入文件时(比如,tsc),才会启用 tsconfig.json。
    推荐使用:tsconfig.json 配置文件。

React 中的常用类型

React 是组件化开发模式,React 开发主要任务就是写组件

两种组件:1 函数组件 2 class 组件。

函数组件,主要包括以下内容:

  • 组件的类型
  • 组件的属性(props)
  • 组件属性的默认值(defaultProps)
  • 事件绑定和事件对象

函数组件

函数组件创建

注意,react的组件,用的是.tsx文件,如果只用.ts文件,是用不了H5的组件的!

一定一定要看一下这个文章:React+TS工程初始化的问题

javascript 复制代码
import React, { FC } from "react";
import ReactDOM from "react-dom";

//定义一个Props的接口
interface Props {
    name: string;
    age: number;
}

//Hello组件,允许传入Props类型参数
//其实不写React.FC也可以。React.FC表示:React.Function Component。
//React.FC 显式地定义了返回类型,作为一个组件返回,其他方式是隐式推导的。
const Hello: React.FC<Props> = ({ name, age }) => (
    <div>
        名字{name}
        年龄{age}
    </div>
)
//Test组件,没有参数
const Test: React.FC = () => (
    <div>
        这里是一个Test组件
    </div >
)

//创建App作为整个页面的基础
const App = () => {
    return (
        <div>
            {/* 组件传值 */}
            <Hello name={'zs'} age={18} />
            <Test></Test>
        </div>
    )

}
//将<App />渲染到root上
ReactDOM.render(<App />, document.getElementById('root'))

关于那个传参的,还是可以再简化一下

函数组件属性的默认值(defaultProps)

第一种写法,有React.FC的显式标注,比较麻烦,换第二种

javascript 复制代码
// 比如我想给组件的某个属性赋一个固定值
const Hello: React.FC<Props> = ({ name, age }) => (
    <div>
        名字{name}
        年龄{age}
    </div>
)
//赋默认值
Hello.defaultProps = {
    age: 18
}

第二种写法,直接在参数列表上赋值,完全按照函数在 TS 中的写法:

javascript 复制代码
//直接在参数上赋值
const Hello = ({ name, age = 18 }: Props) => (
    <div>
        名字{name}
        年龄{age}
    </div>
)

事件绑定和事件对象

事件绑定和事件对象

javascript 复制代码
//Test组件,没有参数
const Test: React.FC = () => (
    <div>
        {/* 绑定点击操作 */}
        <button onClick={onclick}>点击</button>
    </div >
)
//绑定事件
const onclick = () => {
    console.log("点击操作")
}
//这里绑定的是鼠标事件操作
const onclick1 = (e: React.MouseEvent<HTMLButtonElement, MouseEvent>) => {
    console.log("点击操作")
}

//创建App作为整个页面的基础
const App = () => {
    return (
        <div>
            <Test></Test>
        </div>
    )
}
//将<App />渲染到root上
ReactDOM.render(<App />, document.getElementById('root'))

绑定事件写法
再比如,文本框:

javascript 复制代码
//Test组件,没有参数
const Test: React.FC = () => (
    <div>
        {/* 绑定修改输入框操作 */}
        <input onChange={onchange} />
    </div >
)
//这里绑定的是修改事件操作
const onchange = (e: React.ChangeEvent<HTMLInputElement>) => {
    console.log("输入框发生变化")
}

查看可以绑定的事件

实际操作:

确定好可以绑定的事件之后,就可以进行事件绑定

javascript 复制代码
//这里绑定的是输入变化操作
const onchange = (e: React.ChangeEvent<HTMLInputElement>) => {
    console.log("输入框发生变化")
}

class组件

class 组件,主要包括以下内容:

组件的类型、属性、事件 (props)

组件状态(state)

class组件属性(props)

Props(属性):

props 是 React 组件的一种机制,用于向组件传递数据。它是从父组件传递给子组件的数据,而子组件不能直接修改 props,只能读取其中的数据。因此,props 是用于组件之间通信 的一种方式。

在使用组件时,可以在组件标签上添加属性,这些属性将被封装成 props 对象传递给组件。在组件内部,通过解构或直接访问 props 对象,可以获取传递的数据,然后在组件中使用这些数据。

javascript 复制代码
//定义一个Props的接口
type A = {
    name: string;
    age?: number;
}
// React.Component<Props,State>,这里只传入Props不传入State
class Test extends React.Component<A, {}> {
    //将A的age属性赋默认值
    static defaultProps: Partial<A> = {
        age: 18
    }
    render() {
        //通过this.props获取值,当然,也可以在这上面直接赋值
        const { name, age = 20 } = this.props;
        return <><div>名字{name},年龄{age}</div></>
    }
}

//创建App作为整个页面的基础
const App = () => {
    return (
        <div>
            <Test></Test>
        </div>
    )

}
//将<App />渲染到root上
ReactDOM.render(<App />, document.getElementById('root'))

class组件状态(state)

State(状态):

state 是 React 组件用于管理自己的内部状态的一种机制。通过使用 useState 或 useReducer 等 React 提供的钩子或类组件的 setState 方法,可以在组件内部创建和管理状态。

与 props 不同,state 是组件私有的,只能在组件内部访问和修改。当 state 发生改变时,React 将会自动更新组件,并重新渲染显示新的状态。

javascript 复制代码
//定义State名字的type组件
type State = {
    count: number
}

class Counter extends React.Component<{}, State>{
    //将State赋默认值
    static: State = {
        count: 20
        //number: 30 因为没有State没有number,所以无法赋值
    }
    //定义方法,state是组件内部使用的
    onIncrement = () => {
        this.setState({
        	//改变count的值
            count: this.state.count + 1
        })
    }
    render() {
        //这里定义返回组件,调用组件定义的自增函数
        return <><div><button onClick={this.onIncrement}>+1</button></div></>
    }
}

//创建App作为整个页面的基础
const App = () => {
    return (
        <div>
            <Counter></Counter>
        </div>
    )
}
//将<App />渲染到root上
ReactDOM.render(<App />, document.getElementById('root'))

总结React.Component 的 Props和State

props 是组件之间进行数据传递的一种机制,用于从父组件向子组件传递数据。

state 是组件内部维护的状态,用于管理组件的变化和更新。

props 是只读的,组件不能直接修改传递给它的 props。

state 是可变的,可以通过特定的方法修改组件的 state,触发组件的重新渲染。

=======================================================

对于React.Component来说,参数是可选的

定义两个参数type

javascript 复制代码
type State = {
    count: number
}
type Props = {
    message: string
}
相关推荐
w(゚Д゚)w吓洗宝宝了2 小时前
单例模式 - 单例模式的实现与应用
开发语言·javascript·单例模式
zhaocarbon2 小时前
VUE elTree 无子级 隐藏展开图标
前端·javascript·vue.js
小周不摆烂3 小时前
探索JavaScript前端开发:开启交互之门的神奇钥匙(二)
javascript
匹马夕阳5 小时前
Vue 3中导航守卫(Navigation Guard)结合Axios实现token认证机制
前端·javascript·vue.js
我想学LINUX6 小时前
【2024年华为OD机试】 (A卷,100分)- 微服务的集成测试(JavaScript&Java & Python&C/C++)
java·c语言·javascript·python·华为od·微服务·集成测试
screct_demo6 小时前
詳細講一下在RN(ReactNative)中,6個比較常用的組件以及詳細的用法
javascript·react native·react.js
CodeClimb12 小时前
【华为OD-E卷 - 第k个排列 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
光头程序员14 小时前
grid 布局react组件可以循数据自定义渲染某个数据 ,或插入某些数据在某个索引下
javascript·react.js·ecmascript
limit for me14 小时前
react上增加错误边界 当存在错误时 不会显示白屏
前端·react.js·前端框架
浏览器爱好者14 小时前
如何构建一个简单的React应用?
前端·react.js·前端框架