JavaScript无原生不可变类型,需通过约定或工具实现;推荐Immer(语法可变、产出不可变)、次选Immutable.js(功能强但渐进淘汰),简单场景可用Object.freeze或展开运算符。
JavaScript 本身没有原生的不可变数据类型,但可以通过约定、工具方法或专门库来模拟和保障不可变性。核心思路是:不直接修改原始对象或数组,而是每次“变更”都返回一个新副本。
用 Object.freeze、Object.assign、展开运算符(...)、map/filter/concat 等纯函数式方法可以避免直接修改。
Object.freeze(
obj) 只冻结第一层,嵌套对象仍可变,适合简单常量场景{ ...original, key: newValue }
[...arr.slice(0, i), newValue, ...arr.slice(i + 1)] 或用 map
Immer 允许你用“看似可变”的语法编写逻辑,内部自动产出不可变新对象,学习成本低、调试友好、性能优化好。
produce(original, draft => { draft.x = 5; }) —— 写起来像改原对象,实则返回新对象produce 回调里解构赋值或提前 return 原始引用,会绕过代理机制由 Facebook 推出,提供 List、Map、Set、Record 等专属不可变集合类型,带丰富 API 和持久化结构优化。
is() 深比较Immutable.Map()),与原生数组/对象不兼容,调试时显示为自定义类实例如果只需要简单工具函数,可考虑:
immer 已足够覆盖 90% 场景,无需额外引入rfx 或 updeep 提供声明式更新(如 u({ a: { b: 2 } }, state)),适合配置驱动更新
lodash/fp 的 set、update 等函数也支持不可变路径更新,但需注意它默认不深克隆原型链基本上就这些。日常开发推荐从 Immer 入手,兼顾可读性、安全性和迁移成本。真正需要极致性能或历史项目兼容时,再评估 Immutable.js 或定制方案。