ts - 泛型

简介: 泛型(Generics)是指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定类型的一种特性。

泛型

泛型(Generics)是指在定义函数、接口或类的时候,不预先指定具体的类型,而在使用的时候再指定类型的一种特性。

简单的例子

首先,我们来实现一个函数 createArray,它可以创建一个指定长度的数组,同时将每一项都填充一个默认值:

function createArray (len: number, value: any): any[] {
   
  const arr: any[] = []
  for (var i = 0; i < len; i++) {
   
    arr[i] = value
  }

  return arr
}

createArray(3, 'a') // ['a', 'a', 'a']

上例中,我们使用了之前提到过的数组泛型来定义返回值的类型。

这段代码编译不会报错,但是一个显而易见的缺陷是,它并没有准确的定义返回值的类型:

Array<any> 允许数组的每一项都为任意类型。但是我们预期的是,数组中每一项都应该是输入的 value 的类型。

这时候,泛型就派上用场了:

// generics.ts

// function createArray (len: number, value: any): any[] {
   
//   const arr: any[] = []
//   for (var i = 0; i < len; i++) {
   
//     arr[i] = value
//   }
//   return arr
// }

//`Array<any>` 允许数组的每一项都为任意类型。
// 但是我们预期的是,数组中每一项都应该是输入的 `value` 的类型。
// 函数名后添加了 `<T>`,其中 `T` 用来指代任意输入的类型,
// 在后面的输入 `value: T` 和输出 `Array<T>` 中即可使用了
function createArray<T> (len: number, value: T): T[] {
   
  const arr: T[] = []
  for (var i = 0; i < len; i++) {
   
    arr[i] = value
  }
  return arr
}

createArray(3, 'a') // ['a', 'a', 'a']

上例中,我们在函数名后添加了 <T>,其中 T 用来指代任意输入的类型,在后面的输入 value: T 和输出 Array<T> 中即可使用了。

接着在调用的时候,可以指定它具体的类型为 string。当然,也可以不手动指定,而让类型推论自动推算出来:

function createArray<T>(length: number, value: T): Array<T> {
   
  let result: T[] = []
  for (let i = 0; i < length; i++) {
   
    result[i] = value
  }
  return result
}

createArray(3, 'x') // ['x', 'x', 'x']

多个类型参数

定义泛型的时候,可以一次定义多个类型参数: T R U

function swap<T, U>(tuple: [T, U]): [U, T] {
  return [tuple[1], tuple[0]]
}

swap([7, 'seven']) // ['seven', 7]

上例中,我们定义了一个 swap 函数,用来交换输入的元组。

泛型约束

在函数内部使用泛型变量的时候,由于事先不知道它是哪种类型,所以不能随意的操作它的属性或方法:

function loggingIdentity<T>(arg: T): T {
  console.log(arg.length)
  return arg
}

// Property 'length' does not exist on type 'T'.

上例中,泛型 T 不一定包含属性 length,所以编译的时候报错了。

这时,我们可以对泛型进行约束,只允许这个函数传入那些包含 length 属性的变量。这就是泛型约束:

interface Lengthwise {
  length: number
  substr: Function
}

function loggingIdentity<T extends Lengthwise>(arg: T): T {
  console.log(arg.length)
  cosnole.log(arg.substr(1))
  return arg
}
console.log(loggingIdentity('123'))

上例中,我们使用了 extends 约束了泛型 T 必须符合接口 Lengthwise 的形状,也就是必须包含 length 属性。

此时如果调用 loggingIdentity 的时候,传入的 arg 不包含 length,那么在编译阶段就会报错了:

interface Lengthwise {
  length: number
}

function loggingIdentity<T extends Lengthwise>(arg: T): T {
  console.log(arg.length)
  return arg
}

loggingIdentity(7)

// Argument of type '7' is not assignable to parameter of type 'Lengthwise'.

泛型接口

之前学习过接口中函数的定义,可以使用接口的方式来定义一个函数需要符合的形状:

// inter.ts

// 接口的首字母大写
// 一般以I开头
// 接口不是js的对象,元素之间需要使用;隔开而不是,或者是可以不用写
interface IPerson {
   
  firstName: string;
  lastName: string;
}

// function greeter1 (person: IPerson): String {
   
//   return 'hello ' + person.firstName + person.lastName
// }

const greeter1: (person: IPerson) => string = (person: IPerson): string => 'hello ' + person.firstName + person.lastName

const person1 = {
    firstName: '吴', lastName: '大勋' }

greeter1(person1)

当然也可以使用含有泛型的接口来定义函数的形状:

// interGenerics.ts

interface CreateArrayFunc {
   
  <T>(num: number, str: T): Array<T>
}
let createArray1: CreateArrayFunc = <T>(num: number, str: T): T[] => {
   
  const arr = []
  for (let i = 0; i < num; i++) {
   
    // arr.push(str)
    arr[i] = str
  }
  return arr
}
createArray1(5, 'x')

进一步,我们可以把泛型参数提前到接口名上:

// interface CreateArrayFunc {
   
//   <T>(num: number, str: T): Array<T>
// }
// 把泛型参数提前到接口名上, 声明类型注解时 使用 接口<any>
interface CreateArrayFunc<T> {
   
  (num: number, str: T): Array<T>
}
let createArray1: CreateArrayFunc<any> = <T>(num: number, str: T): T[] => {
   
  const arr = []
  for (let i = 0; i < num; i++) {
   
    // arr.push(str)
    arr[i] = str
  }
  return arr
}
createArray1(5, 'x')

注意,此时在使用泛型接口的时候,需要定义泛型的类型。

// src/11generics.ts

// function createArray (num: number, str: any): Array<any> {
   
//   let arr: any[] = []
//   for (let i = 0; i < num; i++) {
   
//     arr[i] = str
//   }
//   return arr
// }


// Array<any> 允许数组的每一项都可以为任意类型
// 实际上输出的数组的类型 就是输入的  str 的数据类型
// 函数名后添加了 <T>,T就用来指代任意输入的类型
// 参数中的 str:T 和返回类型 Array<T>
// function createArray<T> (num: number, str: T): Array<T> {
   
//   let arr: T[] = []
//   for (let i = 0; i < num; i++) {
   
//     arr[i] = str
//   }
//   return arr
// }
// createArray(3, 'a') // T ====》 string
// createArray(3, 5) //   T =====》 number


// 假设如果有多个类型参数呢?
// 定义一个函数,交换输入的元组
// function swap (tuple) {
   
//   return [tuple[1], tuple[0]]
// }
function swap<T, R> (tuple: [T, R]): [R, T] {
   
  return [tuple[1], tuple[0]]
}

swap([7, 'hello']) // ['hello', 7]
swap([true, 9]) // [9, true]

// 泛型约束
// 函数内部使用泛型变量,事先不知 何种类型,不可随意操作他的属性和方法
// function getLen<T> (arg: T): T {
   
//   console.log(arg.length) // 类型“T”上不存在属性“length”。
//   return arg
// }

// 定义时:只允许那些包含了length 属性的变量使用 -- 泛型约束
interface ILength {
   
  length: number
  a: any
  b: any
}
// 丁的泛型 T 必须含有 length 属性了
function getLen<T extends ILength> (arg:T): T {
   
  console.log(arg.length) 
  console.log(arg.a) 
  console.log(arg.b) 
  return arg
}


// 泛型接口
// let createArray1 = <T>(num: number, str: T): T[] => {
   
//   let arr: T[] = []
//   for (let i = 0; i < num; i++) {
   
//     arr[i] = str
//   }
//   return arr
// }


// let createArray1: <T>(num: number, str: T) => T[] = <T>(num: number, str: T): T[] => {
   
//   let arr: T[] = []
//   for (let i = 0; i < num; i++) {
   
//     arr[i] = str
//   }
//   return arr
// }

// interface IFun {
   
//   <T>(num: number, str: T): T[]
// }
// let createArray2: IFun = <T>(num: number, str: T): T[] => {
   
//   let arr: T[] = []
//   for (let i = 0; i < num; i++) {
   
//     arr[i] = str
//   }
//   return arr
// }

// function fn (a) {}
// fn(a)

interface IFun<T> {
   
  (num: number, str: T): T[]
}
let createArray2: IFun<any> = <R>(num: number, str: R): R[] => {
   
  let arr: R[] = []
  for (let i = 0; i < num; i++) {
   
    arr[i] = str
  }
  return arr
}

泛型类

与泛型接口类似,泛型也可以用于类的类型定义中:

class GenericNumber<T> {
   
  zeroValue: T
  add: (x: T, y: T) => T
}

let myGenericNumber = new GenericNumber<number>()
myGenericNumber.zeroValue = 0
myGenericNumber.add = function(x, y) {
    return x + y }

此处 zeroValue,add 未赋值会出错,设置 "strictPropertyInitialization": false, 关闭提示 --- 参照tsconfig.json文件

相关文章
|
弹性计算 Ubuntu Linux
阿里云上搭建Steam游戏【幻兽帕鲁/Palworld】服务器教程
2024年阿里云上搭建Steam游戏【幻兽帕鲁/Palworld】服务器教程,稳定不卡顿。阿里云服务器搭建帕鲁服务器游戏,服务器稳定无卡顿,先下载SteamCMD,并运行;然后下载Palserver,修改服务ini配置,启动PalServer,进入游戏服务器。今天分享阿里云创建幻兽帕鲁服务器教程。
|
前端开发 JavaScript UED
|
8月前
|
JSON JavaScript 前端开发
gRPC技术中的gRPC到HTTP的转换
从上述内容,我们可以看到,gRPC到HTTP的转换并没有改变gRPC强大的性能和可扩展性。它只是让这些强大的功能能更好地适应不同的环境和需求,兼容了更广泛的网络设施。所以,技术总是在变化,并且始终在寻找更好的平衡点,以满足不断变化的业务需求。我们也应该持续学习,掌握这些新的变化,以便更好地解决实际问题。
279 11
|
11月前
|
前端开发 JavaScript UED
React 拖拽排序组件 Draggable List
在现代Web应用中,拖拽排序功能显著提升用户体验。使用React结合`react-dnd`库,可以轻松创建高效且易于维护的拖拽排序组件。通过简单的拖拽操作,用户能直观调整列表项顺序,适用于任务管理、看板工具等场景。实现步骤包括项目初始化、安装依赖、创建基础组件、添加拖拽功能及管理状态和事件。常见问题如拖拽效果不流畅、顺序未更新等可通过性能优化、正确处理索引交换等方式解决。移动端支持也需考虑,确保跨平台的良好体验。
760 25
ts中interface和type的区别
ts中interface和type的区别
904 61
|
12月前
|
JavaScript 前端开发 异构计算
兼容移动手机的js拖拽插件Draggin.js
兼容移动手机的js拖拽插件Draggin.js
317 1
|
JavaScript 前端开发 API
【前端开发】JS同步与异步调用,Vue2基础知识
本文简要介绍了JavaScript中的同步与异步调用以及Vue2的基础知识。 ### JS同步与异步调用 - **同步调用**:代码按顺序执行,每个任务完成后才执行下一个。 - **异步调用**:允许代码并发执行,不必等待前一个任务完成。 - **回调函数**:传统异步模式,如`setTimeout`。 - **Promise**:解决回调地狱问题,链式调用 `.then()`。 - **async/await**:基于Promise,使异步代码看起来像同步代码。 ### Vue2基础知识 - **核心概念**:指令、实例、组件、模板、数据绑定和生命周期钩子。 - **指令**
616 5
|
存储 缓存 资源调度
npm、yarn与pnpm详解
npm、yarn与pnpm详解
428 0
|
JavaScript 安全 API
Vue 3 Composition API 与 Options API:全面比较两者的区别和优缺点
Vue 3 Composition API 与 Options API:全面比较两者的区别和优缺点