美文网首页
TypeScript回调函数该如何声明?

TypeScript回调函数该如何声明?

作者: 陌客百里 | 来源:发表于2020-08-30 23:17 被阅读0次
  • 【序】当我们写Typescript时,我们常常会用到类型声明,但是到定义回调函数时,就用一个any去定义,那么该如何很好的声明回调函数呢?最合理的方法就是使用泛型,所以从这个文章中我们学习到了使用泛型的相关知识*

问题一:

  • 【疑惑】什么是泛型?
  • 【解惑】一种更宽松条件的类型定义,依据你创建变量时定义的类型去推演参数的类型,下列例子
    下面是很常见的使用方法,回调函数中的item是可以动态的追溯到正确的类型的,这是如何实现的呢?
const demo1: Array<number> = [1, 2, 3];
const demo2: Array<string> = ['a', 'b', 'c'];

demo1.map((item) => item);
demo2.map((item) => item);

例如:array的对象如何兼容两种不同的类型呢?如果不使用泛型的话,map的回调函数的参数里就只能用“或”符号来穷举类型。

interface array<number | string>{
  map(callbackfn: (value: string | number, index) => string[]|number[]):string[]|number[]
}

但是上面这种写法过于繁琐,因此可以使用泛型来解决问题,下面是使用泛型的写法

interface array<T>{
  map(callbackfn:(value:T,index:number,array:T[])=> U):U[]
}

此处的关键:首先我们定义了第一个类型变量T,因为在callbackfn中需要生成一个新的类型,这个类型我们不确定,因此新定义了另一个类型变量U

  • 【疑惑】:为什么又T和U?其实没有具体约束,一般约定使用一个大写的字母作为类型变量名

问题二

  • 【疑惑】:泛型还可以怎么用呢?
  • 【解惑】:可以在接口,类,函数中使用
    -- 函数中使用
// 声明一个泛型变量
function identity<T> {}

// 在参数中使用泛型变量
function identity<T>(arg: T) {}

// 在返回值中使用泛型变量
function identity<T>(arg: T): T {}

// 变量声明函数的写法
let myIdentity: <T>(arg: T) => T = identity;

-- 接口中使用

// 使用接口约束一部分数据类型,使用泛型变量让剩余部分变得灵活
interface Parseer<T> {
  success: boolean,
  result: T,
  code: number,
  desc: string
}

// 接口泛型与函数泛型结合
interface Array<T> {
  map<U>(callbackfn: (value: T, index: number, array: T[]) => U): U[]
}

-- 类中使用

// 注意总结相似性
declare namespace demo02 {
  class GenericNumber<T> {
    private value: T;

    public add: (x: T, y: T) => T
  }
}

// 多个泛型变量传入
declare namespace demo02 {
  class Component<P, S> {
    private constructor(props: P);
    public state: S;
  }
}
  • 【疑惑】:在类中和函数中我们比较容易理解泛型,但是在interface中表示了什么意思呢?
  • 【解惑】:我们此时可以将interface看作一个用于定义类型的函数,当在实际使用时,传入什么,就给内部的属性什么类型,可以以这样的想法去看待interface回看下例子就更容易理解。

问题三

  • 【疑惑】:另一个比较常见的场景就是Promise返回一个对象,我们该如何做呢?
  • 【解惑】:要理解类型的嵌套
  • 【关键】:Promise的interface需要一个类型变量,实际调用时我们传入了Result这个类型变量,而这个Result类型变量又需要传入一个类型变量,然后我们又传入Person这个类型变量
interface Person {
  name: string,
  age: number
}

function fetchData(): Promise<Result<Person>> {
  return http.get('/api/demo/person');
}

如果抛开真实的Promise的定义,上面的Promise就可以定义为

interface Promise<T>{
results:T
}
interface Result<T>{
results:T
}

我们实战一下,返回一个带分页的数据

interface Person {
  name: string,
  age: number
}

interface Page<T> {
  current: number,
  pageSize: number,
  total: number,
  data: T[]
}

function fetchData(): Promise<Result<Page<Person>>> {
  return http.get('/api/demo/page/person');
}

【引申】:我们在看一下TS中的数组的定义文件,验证我们的理解

interface Array<T> {
  length: number,
  toString(): string,
  pop(): T | undefined,
  // 注意此处的含义
  push(...items: T[]): number,
  concat(...items: T[]): T[],
  join(separator?: string): string,
  reverse(): T[],
  shift(): T | undefined;
  slice(start?: number, end?: number): T[],
  sort(compareFn?: (a: T, b: T) => number): this,
  splice(start: number, deleteCount?: number): T[],
  // 注意此处的重载写法
  splice(start: number, deleteCount: number, ...items: T[]): T[],
  unshift(...items: T[]): number,
  indexOf(searchElement: T, fromIndex?: number): number,
  lastIndexOf(searchElement: T, fromIndex?: number): number,
  every(callbackfn: (value: T, index: number, array: T[]) => boolean, thisArg?: any): boolean,
  some(callbackfn: (value: T, index: number, array: T[]) => boolean, thisArg?: any): boolean,
  forEach(callbackfn: (value: T, index: number, array: T[]) => void, thisArg?: any): void,
  map<U>(callbackfn: (value: T, index: number, array: T[]) => U, thisArg?: any): U[],
  filter<S extends T>(callbackfn: (value: T, index: number, array: T[]) => value is S, thisArg?: any): S[],
  filter(callbackfn: (value: T, index: number, array: T[]) => any, thisArg?: any): T[],
  reduce(callbackfn: (previousValue: T, currentValue: T, currentIndex: number, array: T[]) => T): T,
  reduce(callbackfn: (previousValue: T, currentValue: T, currentIndex: number, array: T[]) => T, initialValue: T): T,
  reduce<U>(callbackfn: (previousValue: U, currentValue: T, currentIndex: number, array: T[]) => U, initialValue: U): U,
  // reduceRight 略
  // 索引调用
  [n: number]: T,
}

相关文章

  • TypeScript回调函数该如何声明?

    【序】当我们写Typescript时,我们常常会用到类型声明,但是到定义回调函数时,就用一个any去定义,那么该如...

  • 函数指针算法库

    函数指针, 函数指针数组声明 通过参数指定回调函数,代码示例 算法库代码说明 argc:参数个数 argv:回调函...

  • Java进阶之回调机制

    平时经常听别人讲回调函数,听的我一脸懵逼,到底什么是回调函数,回调函数到底该怎么用?下面举例一一说明。 一、回调机...

  • typescript语法

    参考:typescript参考1 typescript参考2 函数参数类型定义 声明函数参数默认值 ...

  • 不懂C语言回调函数,那就看这篇文章吧!

    什么是回调函数 我们先来看看百度百科是如何定义回调函数的: 回调函数就是一个通过函数指针调用的函数。如果你把函数的...

  • TypeSrcipt一起学习(二)

    之前我们学习了TypeScript的数据类型,我们来看看TS中是如何声明函数的 我们先看看js中的函数声明 fun...

  • 异步问题

    什么是回调地狱(函数作为参数层层嵌套)回调函数(一个函数作为参数需要依赖另一个函数执行调用)如何解决回调地狱 pr...

  • TypeScript的函数

    1. 函数声明 1.1 函数声明法 1.2 函数表达式 1.3 匿名函数 TypeScript中的函数也是...

  • JavaScript函数_08回调函数

    回调函数 回调函数(回调),当我们把某个函数作为参数传递给另一个函数的时候,这个函数就是回调函数 回调函数的基本写...

  • Promise

    回调 把一个函数A传给另一个函数B调用,那么A就是回调函数。 回调地狱 回调套回调套回调套回调套回调套回调套回调....

网友评论

      本文标题:TypeScript回调函数该如何声明?

      本文链接:https://www.haomeiwen.com/subject/venesktx.html