typescript泛型枚举以及NaN传染处理

一、泛型

不预先指定具体的类型,而是在使用的时候再指定类型(类型参数化)

语法:使用<T>定义泛型变量

  • T 只是占位符,常用单字母 T(type)K(key)V(value)E(element)P(props)等,如果类型逻辑较复杂,建议使用"以 T 开头的描述性名称",比如:send<TRequest, TResponse>
  • 泛型可以有默认值:<T = unknown>
  • 泛型可以有约束:<T, K extends keyof T> => 表示 K 必须是 T 的键名之一

使用场景:

  1. 在函数上定义的泛型,可以放到函数参数上或作为返回类型,作用:输入输出类型关联
  2. 在 interface 上定义的泛型,可以作为内部子类型,作用:定义通用的对象结构
  3. 在 type 上定义的泛型,作用:创建通用工具类型
  4. 在 class 上定义的泛型,作用:泛型类

示例 1:类型推导

typescript 复制代码
function wrap<T>(value: T): T {
  return value;
}

const s1 = wrap('hello'); // s1: "hello"
const s2 = wrap<string>('hello'); // s2: string

示例 2:通过泛型让外层结构固定,内部数据结构动态变化

typescript 复制代码
interface ApiRes<T> {
  msg: string;
  code: number;
  data: T;
}

interface User {
  id: number;
  name: string;
}

interface Order {
  id: number;
  price: number;
}

function request<T = unknown>(url: string): Promise<ApiRes<T>> {
  return fetch(url).then((res) => res.json());
}

const res1 = await request<User>('/api1'); // res1: ApiRes<User>
console.log(res1.data.name);

const res2 = await request('/api2'); // res2: ApiRes<{ a: 1 }>
console.log(res2.data.a);

示例 3:泛型约束

typescript 复制代码
function getProp<T, K extends keyof T>(obj: T, key: K) {
  return obj[key];
}

const user = { name: '张三' };

const value = getProp(user, 'name'); // value: string

示例 4:通用工具类型

typescript 复制代码
type OrNull<T> = T | null;

const x: OrNull<number> = 1;

二、枚举

使用 as const + keyof typeof 替代 enum

typescript 复制代码
// 1. 定义常量对象
const Colors = {
  RED: 'red',
  BLUE: 'blue',
  GREEN: 'green',
} as const;

// 2. 导出类型
// 解释:
// typeof Colors 获取对象类型
// keyof typeof Colors 获取键 "RED" | "BLUE" | "GREEN"
// typeof Colors[keyof typeof Colors] 获取值 "red" | "blue" | "green"
export type Colors = (typeof Colors)[keyof typeof Colors];

// 3. 使用
function paint(color: Colors) {
  console.log(color);
}

paint('red'); // 可以传字面量
paint(Colors.RED); // 可以传常量
// paint('yellow'); // 报错

三、解决 NaN 的传染性

核心思想:"将运行时的潜在错误,提前到编译时来解决"

NaN 的传染性:如果一个值为 NaN,不管它和谁计算结果还是 NaN,并且 NaN 也不能用"==="检查,所以要避免它在代码中传播

typescript 复制代码
// 解决问题:下行代码没有任何警告,但它结果NaN
Number('a1') + 3;

// 将字符串转换为数字,并给出可能的警告!
function stringToNumber(s: string): number | undefined {
  // Number对不是数字的参数会返回NaN
  const n = Number(s);
  if (isNaN(n)) return undefined;
  return n;
}

// 编译器会警告stringToNumber可能为undefind
const n1 = stringToNumber('a1') + 1;

// 处理警告,从而避免未知错误(正确使用方法)
const n2 = (stringToNumber('a1') ?? 0) + 1;

五、末尾

觉得有帮助可以点点赞。

yaml 复制代码
2025/12/20:发布文章
相关推荐
m0_7482299917 小时前
Vue2 vs Vue3:核心差异全解析
前端·javascript·vue.js
C澒18 小时前
前端监控系统的最佳实践
前端·安全·运维开发
xiaoxue..18 小时前
React 手写实现的 KeepAlive 组件
前端·javascript·react.js·面试
hhy_smile18 小时前
Class in Python
java·前端·python
小邓吖18 小时前
自己做了一个工具网站
前端·分布式·后端·中间件·架构·golang
南风知我意95718 小时前
【前端面试2】基础面试(杂项)
前端·面试·职场和发展
LJianK119 小时前
BUG: Uncaught Error: [DecimalError] Invalid argument: .0
前端
No Silver Bullet19 小时前
Nginx 内存不足对Web 应用的影响分析
运维·前端·nginx
一起养小猫19 小时前
Flutter for OpenHarmony 实战 表单处理与验证完整指南
android·开发语言·前端·javascript·flutter·harmonyos
weixin_3954489119 小时前
main.c_cursor_0130
前端·网络·算法