Symbol 是 JavaScript 的原始类型,用于创建唯一不可变值;可避免属性名冲突、作为内置行为钩子(如 Symbol.iterator),但非私有字段替代品,且 JSON 序列化会忽略。
它不是“对象”,也不是“字符串别名”,而是一种全新的原始类型(和 string、number 并列)。每次调用 Symbol() 都返回一个**全新且不相等**的值,哪怕描述相同:
Symbol('foo') ===
Symbol('foo') // false这种唯一性是核心价值,不是为了“看起来高级”,而是为了解决真实冲突问题。
当多个库、模块或框架往同一个对象上挂方法或属性时,容易无意覆盖。比如两个第三方工具都加 cache 属性,就可能互相破坏。用 Symbol 可彻底隔离:
const cacheKey = Symbol('cache');
obj[cacheKey] = new Map(); // 不会被 obj.cache 或其他 Symbol('cache') 干扰
Symbol.for('cache') 是全局注册表,同名返回同一 Symbol;普通 Symbol('cache') 每次都新造for...in、Object.keys()、JSON.stringify() 遍历或序列化Object.getOwnPropertySymbols() 或 Reflect.ownKeys() 获取JavaScript 引擎在特定操作(如遍历、转字符串)时,会查找对象上是否存在对应名称的 Symbol 方法。这是“魔法”的来源,不是语法糖:
obj[Symbol.iterator] 决定 for...of 怎么遍历它obj[Symbol.toStringTag] 影响 Object.prototype.toString.call(obj) 的输出obj[Symbol.hasInstance] 控制 instanceof 的判定逻辑Symbol 不是私有字段的替代品:
obj[mySym] 仍可读写),只是“隐形”于常规遍历private 或 JS 的 #field 才是真私有obj[mySym],点号 obj.mySym 会找字符串键 "mySym"