TS的重新学习(三)

简介: TS的重新学习(三)

前言


大家好,多日未见,这一次我们继续进行TS的学习,今天讲述的东西可能稍微有些难懂(笔者刚开始学习的时候也是感觉困难)


泛型⚡️⚡️


泛型个人认为是TS中非常重要的一部分,因此我觉得如果泛型学习的差不多,那么TS才叫真正意义上的学好。

这部分内容对于学习过java的同学相信并不陌生


什么是泛型


在学习之前我们先来了解泛型的定义:泛型是指附属于函数,类,接口,类型别名之上的类型,当某个函数的参数,返回值和内部使用的类型无法确定的情况下,就可以使用泛型进行约束。

如何定义泛型呢?

其实也很简单我们可以在函数名称之后加上,这个T就是泛型名称,当然你可以随意的定义名称,我们在向函数传递参数的时候也可以这样传递

举一个例子

function fn<T>(args: T): T {
  return args;
}
fn<string>("hello");

这样T的类型就被限制在字符串类型中,如果我们传递的类型不是string类型的,TS就会报错。


接口中使用泛型


当然我们不仅可以在函数中使用泛型,我们同样可以在接口中定义泛型,我们来看下面的这个例子

interface idfuc<T, Y> {
  id: (value: T) => T;
  name: (value: Y) => Y;
}
let obj2: idfuc<number, string> = {
  id(value) { return value; },
  name(value) { return value; },
};

这里我们在接口中定义使用了泛型,我们进行了类型的约束,这样做的目的就是我们可以提高这个接口的复用性,不事先定义好,而是可以灵活的控制。更加的方便灵活。


类中使用泛型


同样我们在类中也可以使用泛型约束,就比如下面的这个例子:

class Article<T, Y> {
  id: T;
  author: Y;
  constructor(id: T, author: Y) {
    this.id = id;
    this.author = author;
  }
  getAuthor<Y>(name: Y) {
    console.log(name);
  }
}
const article = new Article(12, "de");

这样当我们传入相应的值的时候,TS的类型推断会给我们自动的推断,判断传入的值的类型,这里面我们传递id的值为number类型,传入author的值的类型为string类型,TS就可以自动的给我们推断出来,包括上面举得函数的例子也是如此fn("hello"),如果我们不写,那么TS还是会自动推断为string类型。这就是TS类型推断的好处。


数组泛型


在数组中使用泛型个人认为不多见,下面我们举个例子

let arr: Array<number> = [1, 2, 3, 4] ===  arr:number[]=[1,2,3];


泛型约束🌾🌾


  • 因为在默认情况下T可以代表任何类型,这就导致了无法访问任何属性,所以我们有时候需要进行类型的约束

比如下面这种情况:

function getlength<T>(str:T){
   console.log(str.length);
}
getlength("sdad")

这个时候编译器就会报错,提示T这个类型上面并不存在这个属性,当然我们可以通过判断str的类型来决定是否打印length,但是这样做太过于繁琐,而且使用场景有限,因此泛型约束就应运而生。

那么进行类型约束主要有两种方式:

  • 使用extends关键字进行类型的约束
  • 进行更加具体的类型定义。


extends关键字


注意这里面的extends和我们之前学习的类中的extends不是一个含义,这里的可以理解为约束于,而之前学习的是继承的含义

这里面我们使用第一种方式来进行

function getlength<T extends String>(str: T) {
  console.log(str.length);
}
getlength("sdsd");

如果我们采取这种方式就相当于是将T类型约束为String类型,这样可以获取String上面的所有属性,但同时我们只能传入string类型的值,而且如果我们不需要这么多属性,我们其实可以这样做:

interface IsLength {
  length: number;
}
function getlength<T extends IsLength>(str: T) {
  console.log(str.length);
}

我们可以自己定义接口,然后泛型约束为这个接口以内,这个含义就是传入的参数中的属性必须有length这一项才可以满足需求,这样我们传入数组这种有长度的值的时候也可以满足需求。

第二种方式种提到的更加具体的类型定义是什么意思呢?

比如如果我们想要约束传入的值是一个数组,我们就可以这样做:

function getlength<T>(str:T[]){
   console.log(str.length);
}

这样如果我们传递的不是数组,那么此时就会报错。


keyof关键字


这个部分是对类型约束的补充,也是很重要,那么这个关键字是有什么用呢?

这个关键字可以获取对象中的key值

let obj = {
  name: "sda",
  sex: "男",
};
type Key = keyof typeof obj;

这里就可以获取Key的类型为联合类型,为"name"|"sex",这里面type将他推断为联合类型。

那他的使用场景是什么呢?比如现在我们需要获取一个对象中的某个值,我们就可以使用这个关键字

interface Data {
  name: string;
  age: number;
  sex: string;
}
function ob<T extends Data, K extends keyof T>(obj: T, key: K) {
  return obj[key];
}

这里面我们首先将T约束为Data类型,其次我们这里的key值约束为传递的T中的属性值,这样我们在返回obj[key]的时候才不会报错,否则TS会认为这是不安全的而报错。而且这样做了之后我们在使用函数的时候,传递第二个值会有提示,还是很爽的。


内置工具类型🍆🍆


Record


Record<K extends keyof any, T>这个工具类型的作用是什么呢?

首先我们要明白K extends keyof any的含义,这部分也是用到了类型约束,表明传入的K的类型只能为string|number|symbol 这几个类型中的一个,

用途:Record 工具类型用于创建一个对象类型,其中键为 Key 类型的值,值为 Type 类型的值。

举一个例子:

interface E {
  title: string;
  id: number;
}
type article = "home" | "other" | "about";
type ded=record<article,E>
//等价于
{
  home:{title:string,id:number},
  other:{title:string,id:number},
  about:{title:string,id:number}
}

这样演示一下应该就比较清楚了,其实就是创造了一个对象类型,使用这个工具可以更加的方便,不需要我们在手动的定义了


Partial


Partial

用途:Partial 工具类型将泛型类型的所有属性都变为可选属性。

举一个例子:

interface Person {
  name: string;
  age: number;
}
type PartialPerson = Partial<Person>;
// PartialPerson 的类型是 { name?: string; age?: number; }

这样做就将全部的属性上面加上可选,不需要我们在重复的定义,节省时间,其实内部实现也是比较容易理解的

type Partial<T> = { 
  [P in keyof T]?: T[P];
};

首先通过keyof获取传递进来的对象类型中的属性名,然后通过in关键字进行遍历,其实和for in很像,[[P in keyof T]]代表的就是每次遍历的属性名然后T[P]就是属性值,在加上?就实现了功能


Pick


Pick<T,K>

用途:Pick 工具类型用于从泛型类型中选取指定属性键集合 Keys 对应的属性。

interface Person {
  name: string;
  age: number;
  address: string;
}
type PersonInfo = Pick<Person, 'name' | 'age'>;
// PersonInfo 的类型是 { name: string; age: number; }

从字面上的意思理解就是挑选,其实就是从对象中挑选一部分来作为新的类型。


Omit


Omit<T,K>

用途:Omit 工具类型用于从泛型类型中排除指定属性键集合 Keys 对应的属性。

interface Person {
  name: string;
  age: number;
  address: string;
}
type PersonWithoutAddress = Omit<Person, 'address'>;
// PersonWithoutAddress 的类型是 { name: string; age: number; }

比如有些时候我们可能需要排除一些属性,可以利用大部分属性,又不想要重新在写,这个时候就可以使用这个内置工具类。


小结


今天我们学习了一下泛型以及相关的知识,相信大家还是会有收获的。

相关文章
|
8月前
|
JavaScript 前端开发 API
第9期 一文读懂TS的(.d.ts)文件
第9期 一文读懂TS的(.d.ts)文件
155 0
|
JavaScript Java 编译器
TS的重新学习(二)
TS的重新学习(二)
70 0
|
JavaScript 前端开发 API
一文读懂TS的(.d.ts)文件
一文读懂TS的(.d.ts)文件
4054 0
|
JavaScript 前端开发 开发者
ts知识点
ts知识点
60 0
|
存储 JavaScript Java
TS的重新学习(四)
TS的重新学习(四)
60 0
|
JavaScript 前端开发 Java
对TS的重新学习(一)
对TS的重新学习(一)
53 0
|
JavaScript 前端开发 开发工具
CocosCreator 面试题(五)TS有什么优缺点?为什么要用TS?
CocosCreator 面试题(五)TS有什么优缺点?为什么要用TS?
210 0
|
JavaScript 前端开发 IDE
TS知识点
TS知识点
|
存储 JavaScript 前端开发
重学 TS
重学 TS
174 0
|
JavaScript 前端开发 安全
TS学习(一):基础使用
TS学习(一):基础使用
TS学习(一):基础使用