记录一次使用mitt库后了解到的type和interface异同

起因是在初始化mitt实例时,发现如果传入interface作为泛型时会抛出异常:

类型"Events"不满足约束"Record<EventType, unknown>"。

类型"Events"中缺少类型"string"的索引签名。ts(2344)。

于是去看了一下传入的泛型Event 的定义:

typescript 复制代码
export declare type EventType = string | symbol;
//  ...
export default function mitt<Events extends Record<EventType, unknown>>(all?: EventHandlerMap<Events>): Emitter<Events>;

这里mitt对传入的泛型Events用extends关键字做了个约束,要求Events必须满足Record<EventType, unknown>,也就是说Events必须是一个键为字符串字面量,值为unknownRecord类型。

其中type有隐式索引签名(implicit index signature)Record<PropertyKey, unknown> 自动满足条件,但是interface是一种静态的类型声明,它并没有隐式索引签名,因此会抛出异常,如果需要使用interface,需要手动添加索引签名:

typescript 复制代码
interface Events {
	// ...
	[key: string | symbol]: unknown
}

但是如果这样写的话,会出现以下问题:

  • 接口就的类型约束就被破坏了,key可以是任意的字符串,无法确保事件的key是上面编写的字符串字面量;
  • 接口可以被任意扩展,可能添加非法事件;
  • 无法利用类型系统进行静态检查。
typescript 复制代码
interface Events {
  // ...
}

interface Events {
  [key: number]: string // 新增了数字key
}

为什么mitt要用Record<EventType, unknown>来做限制而不是Record<EventType, any>

unknownany 虽然都代表"不确定的类型",但它们在类型安全性上有着本质区别:

  1. unknown 是类型安全的顶级类型

    • 它是 TypeScript 3.0 引入的类型安全的"任意值"表示
    • 任何值都可以赋给 unknown 类型变量
    • 但在使用前必须进行类型检查或类型断言
  2. any 是类型系统的逃生舱

    • 它会完全绕过 TypeScript 的类型检查
    • 可以对其执行任何操作而不触发类型错误
    • 相当于回到了纯 JavaScript 的开发模式

参考资料:

www.totaltypescript.com/type-vs-int...

相关推荐
Bolt21 小时前
TypeScript 7.0 来了:当 tsc 用 Go 重写之后
javascript·typescript·go
Flynt1 天前
装上TypeScript 7.0 RC之后,最让我意外不是10倍提速
typescript·visual studio code
疯狂SQL1 天前
手写高性能在线 JSON 工具|Web Worker 工程化打包 + 语法自动修复 + 多语言代码生成实战
typescript·json·next.js·web worker·前端性能优化·esbuild·源码实战
Momo__5 天前
TypeScript NoInfer<T>——精准控制泛型推断的工具类型
前端·typescript
退休倒计时6 天前
【每日一题】LeetCode 146. LRU 缓存 TypeScript
算法·leetcode·缓存·typescript
kyriewen7 天前
TypeScript 高级类型:我用 infer 写了一个类型安全的 EventBus,终于搞懂了泛型约束
前端·javascript·typescript
月光刺眼7 天前
Bun + TypeScript 后端入门:从类型约束到 LLM API 调用
后端·typescript
天蓝色的鱼鱼7 天前
Node.js 现在能直接跑 TypeScript 了,tsx 和 ts-node 还需要吗?
前端·typescript·node.js
Oo9207 天前
Bun:下一代 JavaScript/TypeScript 运行时,从入门到实践
typescript·bun
Asize8 天前
Bun + TypeScript 实战:从接口约束到 RESTful 路由设计
后端·typescript·代码规范