TypeScript实战系列之强力爆破泛型的困扰

目录

  • 介绍
    • 开始
    • 如何理解泛型语法
    • 泛型约束
    • 泛型默认值
    • 练习
    • 后续

介绍

泛型在typescript 中使用频率相当高,也给了初学者相当大的阻碍。希望这一篇文章,能帮助你们爆破它。

开始

下面通过模拟实现一个简易版本的axios来引入泛型的使用

// axios.ts
type Method =| 'get'| 'GET'| 'delete'| 'Delete'| 'head'| 'HEAD'| 'options'| 'OPTIONS'| 'post'| 'POST'| 'put'| 'PUT'| 'patch'| 'PATCH'interface AxiosRequestConfig {baseURL?: string //便于为 axios 实例的方法传递相对 URLurl?: string // 请求地址 /* 注 ? 表示可选 */method?: Method // 请求方法 可选data?: any // post、patch 等类型请求的数据 可选params?: any // post、patch 等类型请求的数据 可选headers?: anyresponseType?: XMLHttpRequestResponseType
}type AxiosResponse= {/*** 响应的data数据*/data: any/*** 响应的状态码*/status: number}type AxiosPromise = Promise<AxiosResponse>class Axios {default:AxiosRequestConfig={}constructor(initConfig: AxiosRequestConfig) {console.log('axios---init');}get(url: string, config?: AxiosRequestConfig): AxiosPromise{return new Promise((resolve)=>{resolve({data: {},status:200,})})}
}export const createInstance=(config: AxiosRequestConfig)=>{const instance = new Axios(config)return instance
}

用它来模拟请求

import {createInstance} from "./axios"const axios = createInstance({baseURL:'https://some-domain.com/api/'
})axios.get('user/list').then(res=>{const {data} = res
})

痛点
这时候 我们使用data 里的数据 是没有代码提示的,因为现在data的类型就是any。
在这里插入图片描述

这肯定对于我们开发来说是不方便的!

预期

对于不同的请求,返回的数据格式是不同的。希望每个请求开发者都能自己去定义接口的请求类型!

解决

使用泛型就能完美解决这个痛点!

改造 :

type Method =| 'get'| 'GET'| 'delete'| 'Delete'| 'head'| 'HEAD'| 'options'| 'OPTIONS'| 'post'| 'POST'| 'put'| 'PUT'| 'patch'| 'PATCH'interface AxiosRequestConfig<T=unknown> {baseURL?: string //便于为 axios 实例的方法传递相对 URLurl?: string // 请求地址 /* 注 ? 表示可选 */method?: Method // 请求方法 可选data?: T // post、patch 等类型请求的数据 可选params?: any // post、patch 等类型请求的数据 可选headers?: anyresponseType?: XMLHttpRequestResponseType
}type AxiosResponse<T=unknown>= {/*** 响应的data数据*/data: T/*** 响应的状态码*/status: number}type AxiosPromise<T> = Promise<AxiosResponse<T>>class Axios {default:AxiosRequestConfig={}constructor(initConfig: AxiosRequestConfig) {console.log('axios---init');}get<T=unknown>(url: string, config?: AxiosRequestConfig): AxiosPromise<T>{return new Promise((resolve)=>{resolve({//@ts-ignoredata: {},status:200,})})}
}export const createInstance=(config: AxiosRequestConfig)=>{const instance = new Axios(config)return instance
}

使用:

import {createInstance} from "./axios"const axios = createInstance({baseURL:'https://some-domain.com/api/'
})type UserType = {name:string,age:number
}axios.get<UserType>('user/list').then(res=>{const {data} = res
})

在这里插入图片描述
在这里插入图片描述
总结: 泛型是一种用于创建可复用的函数或类的技术,它允许你在编写代码时指定类型参数,以使函数或类适用于不同类型的数据。

如何理解泛型语法

function identity<T>(arg: T): T {return arg;
}

看到 语法会感到陌生。它没有什么特别,可以把它类比于函数的入参,你可以随意去定义它的参数名,但是一般ts规范使用 大写的英文字母。
在这里插入图片描述
例如:T 就是接受的类型 ,T可以在。只有在外部使用它时 ,T的类型才被确定!

function identity<T>(arg: T): T {let temp:T;let str  = '123'str as Treturn arg;
}

也可以接受多个外部类型

function cross<T,K>(arg1: T,arg2:K): T&K {return {...arg1,...arg2};
}

泛型约束

想要限制外部传入类型必须带有 length 属性,可以使用关键字 extends

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

在这里插入图片描述
改造后

外部传入的类型 必须带有length 属性 ,否则校验不通过

function loggingIdentity<T extends {length:number}>(arg: T): T {console.log(arg.length);return arg;
}loggingIdentity({a:12,length:22})

泛型默认值

泛型也支持给默认值


type arrType<T=string> = {list:Array<T>
}let obj:arrType ={list: ['2','3']
}let obj1:arrType<number> ={list: [1,2]
}

练习

实现一个函数 来通过属性值来获得对应的值

function getValueByKey(obj:any,key:string){return obj[key]
}let obj = {name:'kd',age:35
}let value = getValueByKey(obj,'name')

在这里插入图片描述

但是现在没有代码提示,需要使用泛型,一个代表传入对象的类型,一个代表传入key 的类型


function getValueByKey<T,K>(obj:T,key:K):T[K]{return obj[key]
}

在这里插入图片描述
因为 ts 不知道 T 和 K 的 关系 所以会报错,但是我们开发知道他们的关系 K 是 属于 T的 key值,于是我们给他们加上 类型限制!


function getValueByKey<T extends {[P:string]:any},K extends keyof T>(obj:T,key:K):T[K]{return obj[key]
}

keyof 关键字 后面再介绍, 它的作用是取 类型的 key 值的!
在这里插入图片描述
代码提示就出来了! 至于为啥使用的时候可以省略尖括号 是因为类型推断
在这里插入图片描述
这时候 T 代表 {name:string,age:number} K 代表 “name” | “age”

当然你也可以自己指定

function getValueByKey<T extends {[P:string]:any},K extends keyof T>(obj:T,key:K):T[K]{return obj[key]
}let obj = {name:'kd',age:35
}let value = getValueByKey<typeof obj, keyof typeof obj>(obj,'age')

后续

后面再会出一篇ts的高级篇,里面会涉及到更多的泛型的使用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/657097.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

如何在centos7上配置为桥接模式

一、打开虚拟机的设置页面&#xff0c;设置虚拟机桥接模式如图&#xff1a;选择桥接模式&#xff08;复制物理网络连接可选&#xff09; 二、net0对应桥接模式的配置&#xff0c;如下方式选择 三、 在 CentOS 7 中&#xff0c;通过编辑网络配置文件来配置网络参数。找到 /etc…

(九)springboot实战——springboot3下的webflux项目参数验证及其全局参数验证异常处理

前言 在上一节内容中&#xff0c;我们介绍了如何在webflux项目中自定义实现一个全局的异常处理器ErrorWebExceptionHandler&#xff0c;正常情况下其可以处理我们系统的运行时异常&#xff0c;但是无法处理参数验证的异常WebExchangeBindException&#xff0c;所以这里提供另外…

算法随想录第四十八天打卡| 198.打家劫舍 , 213.打家劫舍II , 337.打家劫舍III

详细布置 今天就是打家劫舍的一天&#xff0c;这个系列不算难&#xff0c;大家可以一口气拿下。 198.打家劫舍 视频讲解&#xff1a;动态规划&#xff0c;偷不偷这个房间呢&#xff1f;| LeetCode&#xff1a;198.打家劫舍_哔哩哔哩_bilibili 代码随想录 class Solution(…

STM32 E18-D80NK红外避障传感器

E18-D80NK-N是一款红外光电传感器&#xff0c;它同时具备发射和接收功能。通过对发射光进行调制后发出&#xff0c;并通过接收头对反射光进行解调输出。 E18-D80NK-N采用了透镜来增强传感器的性能&#xff0c;使其能够检测更远的距离。根据红外光的特性&#xff0c;不同颜色的…

拒掉了一个双 985 的面试者

下班路上&#xff0c;和一个大佬聊天&#xff0c;他说今天面试&#xff0c;拒掉了一个双 985 的候选人。 候选人背景很好&#xff0c;本科和硕士都是 985 院校毕业&#xff0c;并且是除了清北浙之外&#xff0c;排名非常靠前的 985院校。 本硕专业都是计算机专业&#xff0c;…

Spring Boot + security + jwt 测试安全策略

一、测试概述 主要目的是测试security的用法。因测试搭建mysql和redis比较麻烦&#xff0c;所以我这里将自定义的jwt和用户信息缓存到程序的内存中。 本人测试的项目比较混乱&#xff0c;Spring Boot父类只标出有用的依赖。其子类用的版本为jdk11。后续会继续深入oauth2&#x…

数据库之TiDB基础讲解

文章目录 1 TiDB1.1 引言1.2 TiDB介绍1.3 系统架构1.3.1 TIDB Server1.3.2 PD Server1.3.3 TIKV Server1.3.4 TiKV如何不丢失数据1.3.5 分布式事务支持 1.4 与MySQL的对比1.5 性能测试1.5.1 测试一1.5.2 系统测试报告 2 1 TiDB 1.1 引言 当我们使用 Mysql 数据库到达一定量级…

调试小结:PHY初始化前后Link Status是否能正确反应网线插上、拔下状态

1 说明 为了验证是否需要初始化PHY才能检测到网线插上、拔下&#xff0c;这里我们对比初始化PHY&#xff08;LAN8720&#xff09;前后&#xff0c;插拔网线PHY寄存器1的bit2的是否按照预期变化来进行测试。 我们查看的PHY寄存器是1的bit2&#xff0c;定义如下&#xff1a; 2…

springboot的actuator

1、actuator简介 微服务的特点决定了功能模块的部署是分布式的&#xff0c;大部分功能模块都是运行在不同的机器上&#xff0c;彼此通过服务调用进行交互&#xff0c;前后台的业务流会经过很多个微服务的处理和传递&#xff0c;出现了异常如何快速定位是哪个环节出现了问题&am…

STM32 CAN接口中断处理与性能优化技巧

在基于STM32的CAN接口中&#xff0c;中断处理是一个非常重要的部分&#xff0c;它可以帮助我们实时地处理接收到的数据和处理其他CAN事件。为了优化CAN接口的性能&#xff0c;以下是一些中断处理和性能优化的技巧&#xff1a; ✅作者简介&#xff1a;热爱科研的嵌入式开发者&am…

系统架构19 - 面向对象

面向对象设计 相关概念面向对象分析基本步骤基本原则分析模型 面向对象设计设计模型类的类型 面向对象编程基本特点需求建模设计原则面向对象软件测试 相关概念 接口&#xff1a;描述对操作规范的说明&#xff0c;其只说明操作应该做什么&#xff0c;并没有定义操作如何做。消…

C与C++相互调用的“底层原理“

在你的C语言代码中&#xff0c;不知能否看到类似下面的代码&#xff1a; 这好像没有什么问题&#xff0c;你应该还会想&#xff1a;“嗯⋯是啊&#xff0c;我们的代码都是这样写的&#xff0c;从来没有因此碰到过什么麻烦啊&#xff5e;”。 你说的没错&#xff0c;如果你的头…

ES6理论及方法

一、基础知识 1.不可枚举就是不可遍历。 2.数据代理&#xff1a;通过一个对象代理对另一个对象中属性的操作&#xff08;读/写&#xff09; 二、方法 1.一个字符串是否包含在另一个字符串中 includes()&#xff1a;返回布尔值&#xff0c;表示是否找到了参数字符串。 starts…

Excel中将16进制数转化成10进制(有/无符号)

Excel中将16进制数转化成10进制&#xff08;有/无符号&#xff09; Excel或者matlab中常用XXX2XXX进行不同进制的转换 16进制转10进制&#xff08;无符号数&#xff09;&#xff1a;HEX2DEC 16进制转10进制&#xff08;有符号数&#xff09;&#xff1a; FA46为例&#xff0c…

看到小米SU7这顶配价格 全都坐不住了

文 | AUTO芯球 作者 | 李诞 果然被我猜中了吗&#xff1f; 我是真的猜中了吗 之前我的视频里说 小米SU7这技术、这车 绝对值35万 今天爆出来 小米汽车上牌售价 保险费用显示顶配36.14万 难怪雷总说 这是按照Dream Car的标准来设计的 投入百亿 雷总这点是真没骗人 …

架构整洁之道-价值维度与编程范式

1 设计与架构究竟是什么 结论&#xff1a;二者没有任何区别&#xff0c;一丁点区别都没有。 架构图里实际上包含了所有底层设计细节&#xff0c;这些细节信息共同支撑了顶层的架构设计&#xff0c;底层设计信息和顶层架构设计共同组成了整个架构文档。底层设计细节和高层架构信…

Matplotlib应用-股票技术分析实战

MACD Moving Average Convergence/Divergence&#xff0c;意为异同移动平均线。它刻画的是股价变化的速度 MACD算法 指标含义公式短期EMA短期收盘价指数移动均线(12天)前一日EMA(12)11/13 今日收盘价2/13长期EMA长期收盘价指数移动均线(26天)前一日EMA(26)25/27 今日收盘价2…

数据结构——并查集

1.并查集的定义 并查集其实也是一种树形结构&#xff0c;在使用中通常用森林的方式来表示 并查集的逻辑结构其实就是集合 并查集一般可以通过双亲写法&#xff08;顺序结构&#xff09;来完成&#xff0c;即通过一个数组存储父亲结点的下标 int s[10005]; int main() {for(…

深入了解Yum:Linux系统的软件包管理利器

目录 软件包 软件包的来源 关于yum yum是什么 yum的相关操作 介绍rzsz rz&#xff08;从Windows本地传到Linux服务器&#xff09; sz&#xff08;从Linux服务器传到Windows本地&#xff09; 注意事项 查看软件包 安装软件 卸载软件 yum的本地配置 为什么要进行配置…

《基于“源启+”的应用重构白皮书》

当前&#xff0c;行业数字化转型驶入“深水区”&#xff0c;全新的市场竞争格局对行业发展提出更高的要求&#xff0c;企业高质量发展需要借助新架构新应用重新定义数字生产力&#xff0c;重塑商业模式与市场核心竞争力。 在中国电子主办&#xff0c;中电金信承办的“数字原生向…