问: 在 TS 中普通枚举和常量枚举有什么区别?

引言

TS 中枚举(enum)是一个比较常用的语法, 通俗来说, 枚举就是对一个对象的所有可能取到的值的集合, 枚举的定义方式与结构体和联合体相似, 其形式为:

ts 复制代码
enum 枚举名{
  标识符①[=整型常数],
  标识符②[=整型常数],
  ...
  标识符N[=整型常数],

也许你看到过下面这两种枚举(enum)定义的方式:

ts 复制代码
// 普通枚举
enum Color { Red, Green, Blue }

// 常量枚举(使用 const)
const enum Color { Red, Green, Blue }

那么问题来了, 这两种定方式有什么区别呢?

其实上面两种声明方式唯一的区别就在于编译结果上有所不同, 下面我们将对其展开简单说明, 对于 TS 枚举的基本使用, 语法这边就不展开讲解了

一、普通枚举

如下我们使用 普通枚举 的方式, 声明了一个枚举 Color

ts 复制代码
// 普通枚举
enum Color { Red, Green, Blue }

下面我们可以在 Typescript 游乐场 来看下上面 TS 编译后的结果

从图中我们会发现 普通枚举 编译后的 JS 代码是会去创建一个对象 Color, 然后往对象上添加一些列的值。这里我们可以在自己项目中测试下, 打印出 Color 看看它是长啥样的

如上图, 可以看出, 最终编译后生成的 Color 对象实际上是一组 key => value 以及 value => key 的一个映射表

ts 复制代码
// 普通枚举
enum Color { Red, Green, Blue }

Color.Red // 0
Color[2] // "Blue"

对于普通枚举, 最终编译出来的就是一个普通的对象, 如上代码译出来的 Color 就是一个普通的对象, 之后 Color 是有可能被篡改的(尽管它们是只读的, 但我们依然是有办法修改的)

ts 复制代码
// 普通枚举
enum Color { Red, Green, Blue }

Color.Blue; // 2

// Color 中属性都是只读的, 可通过 Object.defineProperty 进行修改
Object.defineProperty(Color, 'Blue', { writable: true, value: 666 });

Color.Blue; // 666
Color.Blue; // 666

二、常量枚举

那么相对于普通枚举(只使用 enum 进行声明), 常量枚举(使用 const + enum 进行声明)在编译结果上又有什么不一样呢?

如下我们使用 常量枚举 的方式, 声明了一个枚举 Color

ts 复制代码
// 常量枚举(使用 const)
const enum Color { Red, Green, Blue }

const red = Color.Red
const red = Color.Green

下面我们可以在 Typescript 游乐场 来看下上面 TS 编译后的结果

TS 编译后的结果会发现, 常量枚举和普通枚举的差异还是很大的, 对于常量枚举会在 TS 编译期间被删除, 常量枚举的成员在使用的地方会被替换为对应的值

三、总结

  1. 普通枚举 enum A {...} 和常量枚举 const enum A {...} 之间的区别主要在于 TS 的编译结果上有所差别
  2. 普通枚举 enum A {...}, 会将其编译为一个 JS 对象, 对象内就是枚举成员和值的一个相互映射
  3. 常量枚举 const enum A {...}, 编译后不会生成任何代码, 会删除 TS 部分内容, 对于使用到的成员只会进行值的替换
  4. 由此可见, 使用 常量枚举 会有更好的性能, 避免额外的性能开销; 那么大部分情况下我们更推荐使用常量枚举, 那么问题来了我们什么时候使用普通枚举呢? 答案是俺也没找到答案😭😭😭😭

四、参考

相关推荐
zhougl9961 小时前
html处理Base文件流
linux·前端·html
花花鱼1 小时前
node-modules-inspector 可视化node_modules
前端·javascript·vue.js
HBR666_1 小时前
marked库(高效将 Markdown 转换为 HTML 的利器)
前端·markdown
careybobo2 小时前
海康摄像头通过Web插件进行预览播放和控制
前端
TDengine (老段)3 小时前
TDengine 中的关联查询
大数据·javascript·网络·物联网·时序数据库·tdengine·iotdb
杉之4 小时前
常见前端GET请求以及对应的Spring后端接收接口写法
java·前端·后端·spring·vue
喝拿铁写前端4 小时前
字段聚类,到底有什么用?——从系统混乱到结构认知的第一步
前端
再学一点就睡4 小时前
大文件上传之切片上传以及开发全流程之前端篇
前端·javascript
木木黄木木5 小时前
html5炫酷图片悬停效果实现详解
前端·html·html5
请来次降维打击!!!6 小时前
优选算法系列(5.位运算)
java·前端·c++·算法