彻底讲透as const + keyof typeof

一、为什么要 "as const"

1、默认情况下,TS把字面量推断成"宽类型"

arduino 复制代码
const a = 'foo' // 类型是 string,不是 'foo'

于是你写 'foo' | 'bar' 时, 一旦拼错成 'fooo', TS不会拦你。

2、加上 as const ,TS会把它当成"不可变的精确字面量"

csharp 复制代码
const a = 'foo' as const // 类型是 'foo',宽不了

数组、对象同理:

csharp 复制代码
const arr = [1, 2, 3] // number[]
const arr2 = [1, 2, 3] as const // readonly [1, 2, 3] // 元组 + 字面量

二、keyof typeof: 把值世界映射到类型世界

  • typeof:能把运行时的值变成类型;
  • keyof:能把对象类型变成联合类型

合起来就是:给我一个对象,我立刻推出它所有建的字面量联合。

ini 复制代码
const obj = {a: 1, b: 2} as const
type keys = keyof typeof obj // 'a' | 'b'

三、组合拳实战:从"一段配置"到"类型+运行时"全打通

需求:做一个下拉框,选项是「所有摄像头名称」,后台接口也只接受这四个字符串;以后加新摄像头,改一处即可

步骤1:写配置(as const 固化)

css 复制代码
// camera.ts
export const cameraMap = {
  student_closeup:  { label: '学生特写', stream: '/stu/close' },
  student_full:     { label: '学生全景', stream: '/stu/full' },
  teacher_closeup:  { label: '老师特写', stream: '/tea/close' },
  teacher_full:     { label: '老师全景', stream: '/tea/full' },
} as const

步骤 2:一次性导出所有要用的类型

typescript 复制代码
// 从同一份配置推导,零手工维护
export type CameraKey   = keyof typeof cameraMap      // 'student_closeup' | ...
export type CameraMeta  = typeof cameraMap[CameraKey] // 每个 value 的结构

步骤3:组件里随便用

csharp 复制代码
/* 下拉框当前值,类型是 CameraKey,写错字母会报错 */
const selected = ref<CameraKey>('student_closeup')
相关推荐
神奇的程序员14 分钟前
开发了一个nginx日志分析面板
前端
阿拉丁的梦25 分钟前
【C4D实用脚本】清除废点及删除了面的选择tag和材质tag及材质--AI编程
服务器·前端·材质
傅里叶27 分钟前
Flutter移动端获取相机内参
前端·flutter
哒哒哒52852031 分钟前
React useMemo 大白话用法文档(含注意项)
前端
xkxnq33 分钟前
第一阶段:Vue 基础入门(第 10 天)
前端·javascript·vue.js
智商偏低33 分钟前
abp PermissionDefinitionManager源码解析
开发语言·前端·javascript
RaidenLiu34 分钟前
Offstage / Visibility:不可见真的就不消耗性能吗
前端·flutter·性能优化
lgliuying35 分钟前
wangEditor5 富文本编辑器中使用 kityformula 公式编辑器的具体实践
前端·javascript·html
Benny的老巢36 分钟前
基于Playwright TypeScript/JavaScript的API调用爬虫成熟方案
javascript·爬虫·typescript·自动化·agent·playwright
PBitW43 分钟前
Electron 脚本调用大坑!害惨我了
前端·electron