typescript笔记
什么是TS
TypeScript是一种由微软开发的自由和开源的编程语言,它是JavaScript类型的超集。
为什么要用TS
- 程序更容易理解(函数或者方法输入输出的参数类型,外部条件等);
- 效率更高,在不同代码块和定义中进行跳转,代码补全等;
- 更少的错误,编译期间能够发现大部分的错误,杜绝一些比较常见的错误;
- 好的包容性,完成兼容 JavaScript,第三方库可以单独编写类型文件;
环境搭建
| |
| npm install -g typescript |
| |
| tsc helloworld.ts |
复制
一般就是项目启动前安装到项目依赖中,构建工具自动编译
常用数据类型约束
ts基本类型约束写法
- 字符串、数值、布尔、any、null、undefined
| |
| let str: string = '你好呀'; |
| |
| let num: number = 123; |
| |
| let bool: boolean = true; |
| |
| let an: any = true; |
| |
| |
| let nu:null = null; |
| |
| let ud:undefined = undefined; |
| |
| let score: number | null = 80; |
复制
数组约束写法
| let arr1: number[] = [1,2,3,4]; |
| let arr2: string[] = ['a','b','c']; |
| |
| let arr3: Array<number> = [1,2,3]; |
复制
| |
| let arr4: (string | number)[] = [1,'b']; |
| |
| let arr5: [number,string] = [1,'b']; |
复制
| |
| const positon:[x,y] = [102.85076697175514, 24.89248383034783] |
复制
对象约束写法
| let obj1:{ |
| a:number; |
| b:number; |
| c:number; |
| } = { |
| a:1, |
| b:2, |
| c:3 |
| } |
复制
| type Obj = { name: string; age?: number }; |
| |
| let obj: Obj = { name: '', age: 18 } |
| |
| let obj1: Obj = { name: '' } |
复制
| type Student = { |
| id: string; |
| name: string; |
| [propName: string]: any |
| } |
| |
| const Stu1: Student = { |
| id: 'sjflsjfdlk', |
| name: 'hr', |
| age: 20 |
| } |
复制
| type Student = { |
| readonly id: string; |
| name: string; |
| } |
| const stu1: Student = { |
| id: 'sjflsjfdlk', |
| name: 'hr', |
| age: 20 |
| } |
| |
| stu1.id = '99999999' |
复制
函数约束
| function fun1( a:number, b: number ): number { |
| return a+b; |
| } |
复制
| function fun1( a:number, b?: number ): number { |
| if( b ){ |
| b = 123; |
| } |
| return a; |
| } |
复制
| function fun3( a:number , b:number=1,c:{ name:string,age?: number } = { name:'wgh' }){ |
| } |
复制
| |
| |
| |
| |
| |
| let fun4:(params1:number,params2:string) => string = (a:number,b:string):string =>{ |
| return a + b; |
| } |
| |
| const fn = (a: string, b: string): string => a + b; |
复制
interface & type
interface
| |
| interface Person { |
| name: string; |
| age: number; |
| } |
| const p: Person = { |
| name: 'wgh', |
| age: 18 |
| } |
| |
| |
| interface Fn { |
| (a, b): string; |
| } |
| const fn: Fn = (a, b) => `${a}+${b}`; |
| |
| |
| interface NumberArr { |
| [index: number]: string | number; |
| } |
| const arr: NumberArr = ['', 6]; |
复制
type
| |
| type name = string; |
| type age = string; |
| |
| type sex = string | number; |
| |
| type Position = [number, number]; |
| |
| type ScoreArr = number[]; |
| |
| |
| type Person = { |
| name; |
| age; |
| } |
| |
| type Fn = (a: number,b: string) => string; |
复制
相同点
| |
| interface Person { |
| name: string; |
| age: number; |
| } |
| const p: Person = { |
| name: 'wgh', |
| age: 18 |
| } |
| |
| |
| interface Fn { |
| (a, b): string; |
| } |
| const fn: Fn = (a, b) => `${a}+${b}`; |
| |
| |
| type Person = { |
| name; |
| age; |
| } |
| |
| type Fn = (a: number,b: string) => string; |
复制
-
都可以被继承
| interface A { |
| name:string |
| } |
| interface B extends A{ |
| age:number |
| } |
| const p: B = { |
| name: '', |
| age: 18 |
| } |
复制
| type A = { |
| name:string |
| } |
| |
| interface B extends A { |
| age: number |
| } |
复制
| type A = { |
| name:string |
| } |
| |
| type B = A & { |
| age: number |
| } |
复制
| interface A { |
| name:string |
| } |
| |
| type B = A & { |
| age: number |
| } |
复制
-
都能实现implements
| |
| interface ICat{ |
| setName(name:string): void; |
| } |
| class Cat implements ICat{ |
| setName(name:string):void{ |
| |
| } |
| } |
| |
| |
| type ICat = { |
| setName(name:string): void; |
| } |
| |
| class Cat implements ICat{ |
| setName(name:string):void{ |
| |
| } |
| } |
| |
| |
| type Person = { name: string } | { setName(name: string): void }; |
| |
| |
| class Student implements Person { |
| name = '张三'; |
| setName(name: string): void { |
| |
| } |
| } |
复制
不同点
| interface A {} |
| type A = {} |
| |
| interface B extends A {} |
| type B = A & {} |
复制
- type可以定义基本数据类型约束
- type可以定义联合类型约束
| type TestUniteType = string | { name: string } |
复制
| type Tuple = [string,string] |
复制
以上都是interface做不到的,接下来记录type做不到的
如果你多次声明一个同名的接口,TypeScript 会将它们合并到一个声明中,并将它们视为一个接口。这称为声明合并。
| interface Person { name: string } |
| interface Person { age: number } |
| |
| let user: Person = { |
| name: "Tolu", |
| age: 0, |
| } |
| |
| type People = { name: string }; |
| type People = { age: number } |
| |
| |
| const p: People = { |
| name: '', |
| age: '', |
| }; |
复制
| |
| interface propType{ |
| [key: string] : string |
| } |
| |
| let props: propType |
| |
| type dataType = { |
| title: string |
| } |
| interface dataType1 { |
| title: string |
| } |
| const data: dataType = {title: "订单页面"} |
| const data1: dataType1 = {title: "订单页面"} |
| props = data |
| |
| props = data1 |
复制
总结
- 官方推荐用 interface,其他无法满足需求的情况下用 type。
- 但其实,因为 联合类型 和 交叉类型 是很常用的,所以避免不了大量使用 type 的场景,一些复杂类型也需要通过组装后形成类型别名来使用。 所以,如果想保持代码统一,还是可选择使用 type。通过上面的对比,类型别名 其实可涵盖 interface 的大部分场景。
- 对于 React 组件中 props及 state,使用 type ,这样能够保证使用组件的地方不能随意在上面添加属性。如果有自定义需求,可通过 HOC二次封装。
- 编写三方库时使用interface,其更加灵活自动的类型合并可应对未知的复杂使用场景。
参考博客
类
定义类
| class Person { |
| userName:string; |
| userAge:number; |
| constructor( name:string, age:number ){ |
| this.userName = name; |
| this.userAge = age; |
| } |
| run( a:number, b:number ):number { |
| return a+b; |
| } |
| } |
| new Person( '张三' , 18 ).run(1,2); |
复制
修饰符
- readonly 只读
- public 公开的,在任何地方都可以访问
- protected 受保护的,只能在当前类和当前类的子类内部使用
- private 私有的,当前类的内部使用
抽象类
不完成具体的功能 抽象类不能new 抽象类可以继承,如果要继承,就必须实现该类的抽象方法
| abstract class Person{ |
| abstract run():void; |
| abstract change():void; |
| } |
| |
| abstract class Person{ |
| abstract run():void; |
| abstract change():void; |
| } |
复制
implements对于类的约束
| interface IAnimal{ |
| name:string; |
| age:number; |
| } |
| |
| interface IAnimalDo{ |
| eat():void; |
| } |
| |
| class Cat implements IAnimal,IAnimalDo{ |
| name: string; |
| age: number; |
| eat():void{}; |
| constructor(name:string,age:number) { |
| this.name = name; |
| this.age = age; |
| } |
| } |
复制
泛型
什么是泛型
是指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定类型的一种特性。
函数的泛型写法
| function fun<T>(arg:T):T{ |
| return arg |
| } |
| fun<string>('wgh'); |
| fun<number>(5); |
| |
| function fun2<T,U>(a:U,b:T) { |
| return `${a},今年+${b}岁`; |
| } |
| fun2<number,string>('wgh',25); |
复制
接口的泛型写法
| |
| interface IData { |
| length:number |
| } |
| |
| function fun5<I extends IData>(arg: T): number { |
| return arg.length; |
| } |
| fun5<number[]>([1,2,3,4,5,6]); |
| fun5<string>('my name is wgh'); |
| fun5<string[]>(['1','2']); |
| fun5<HTMLCollection>(document.querySelectorAll('div')); |
复制
class的泛型写法
| class Person<T,U,I>{ |
| name:T; |
| age:U; |
| fun<I>(){} |
| constructor(name:T,age:U) { |
| this.name = name; |
| this.age = age; |
| } |
| } |
| const p1 = new Person<string,number,number[]>('wgh',25) |
复制
类型断言
- 语法:值 as 类型
- 将一个联合类型断言为其中一个类型
| type Fish = { |
| name: string; |
| swim(): void; |
| } |
| |
| type Cat = { |
| name: string; |
| run(): void; |
| } |
| |
| const fnLogName = (animal:Fish|Cat) => { |
| console.log(animal.name); |
| } |
| |
| const fnDo = (animal:Fish|Cat) => { |
| (animal as Fish).swim() |
| } |
| |
| const tom:Cat = { |
| name:'汤姆', |
| run() { |
| console.log(this.name + '会跑'); |
| } |
| } |
| |
| const swim = (animal:Fish|Cat) => { |
| (animal as Fish).swim(); |
| } |
| |
| swim(tom); |
复制
| class TypeError extends Error{ |
| code: number = 0; |
| } |
| |
| class HttpError extends Error{ |
| statusCode:number = 200; |
| } |
| |
| const isApiError = (code:Error):string|undefined => { |
| if((code as HttpError).statusCode == 200){ |
| return 'apiError'; |
| } |
| } |
复制
复制
| const Luca:any = { |
| name:'路西', |
| run: () => {} |
| }; |
| (Luca as Cat).run(); |
复制
-
类型断言限制
- 联合类型可以被断言为其中一个类型
- 父类可以被断言为子类
- 任何类型都可以被断言为any
- any可以被断言为任何类型
-
双重断言
| let cont: number = 1; |
| (cont as any as string).length; |
| |
复制
命名空间
- ECMAScript 2015开始 在代码量较大的情况下,为了避免各种变量命名相冲突,可将相似功能的函数、类、接口等放置到命名空间内
- 命名空间:内部模块,主要用于组织代码,避免命名冲突。
- 模块: ts的外部模块的简称,侧重代码的复用,一个模块里可能会有多个命名空间。
| namespace B { |
| class Animal {} |
| } |
| namespace B { |
| class Animal {} |
| } |
复制
~## 装饰器~
枚举
用于定义命名常量集合的数据类型。
定义枚举
| |
| enum Color { |
| red, |
| blue, |
| yellow |
| } |
| |
| |
| enum Animal { |
| dog = -99, |
| cat, |
| pig |
| } |
复制
枚举访问
| enum Color { |
| red, |
| blue, |
| yellow |
| } |
| const color = Color.red; |
复制
字符串枚举
| |
| enum Color { |
| RED, |
| BLUE = 'blue', |
| YELLOW |
| } |
复制
操作符和操作符
| |
| const obj: { |
| name?: string; |
| age?: null | { a: number }; |
| [propName: string]: any; |
| } = {}; |
| |
| const b: string = null!; |
| const a: number = obj.age!; |
| |
| obj.age!.a = 6 |
| |
| formSchema.find((item) => item?.props?.field === "weeks")!.props!.columns = semesterCalWeek( |
| startAt, |
| endAt, |
| ); |
复制
| !!0 |
| !!1 |
| !!undefined |
| !!null |
| !!NaN |
| |
| |
| undefined ?? 1 |
| null ?? 1 |
| NaN ?? 1 |
复制
| type a = { |
| name: number |
| } |
| type b = a & { |
| age: number |
| } |
| |
| const a:string | number = 1 |
复制
| const a = 1_0000_0000; |
| console.log(a); |
复制