immer.js 简介及源码简析:更简单,更快速的创建不可变数据类型 · zhao.zhang

阅读时长 3 分钟读完

在前端开发中,不可变数据类型(immutable data)越来越受到关注。使用不可变数据可以避免因为修改引用类型数据而带来的副作用,例如无法正确比较对象或数组是否相等,以及导致组件重复渲染等问题。

然而,在 JavaScript 中创建不可变数据类型通常需要写大量的代码,并且需要非常小心地管理所有的对象和数组。幸运的是,immer.js 可以帮助我们更轻松地创建不可变数据类型。

immer.js 简介

immer.js 是一个 JavaScript 库,它提供了一种简单且直观的方式来创建不可变数据类型。它通过使用“草稿”概念来实现这一点。在草稿中,你可以像修改普通对象一样修改嵌套对象的属性,immer 会自动处理不可变性并返回新的不可变数据。同时,immer 在内部使用了一些性能优化技巧来确保创建不可变数据的速度非常快。

使用 immer.js

让我们看一个示例,假设我们有一个包含商品的列表:

现在我们想要将第一个商品的价格增加 5 元。使用 immer.js,我们可以这样做:

在上面的代码中,produce 函数接受一个原始的对象或数组和一个函数作为参数。该函数会被传递一个“草稿”对象,你可以像修改普通对象一样地修改它。最后,produce 函数将返回一个新的不可变对象或数组。

注意,在草稿中修改的任何对象或数组属性都必须是可变的,否则 immer 会抛出错误。例如,如果你尝试修改一个 const 变量,那么将会引发错误。

源码简析

让我们进一步深入 immer.js 的源码来理解它的实现方式。在 immer.js 中,对象和数组都被称作“producers”,即生产者。每个生产者都有一个“draft”版本和一个“base”版本。草稿版本是可变的,而基础版本是不可变的。当你在草稿中修改对象或数组时,immer 会自动创建一个新的草稿版本,并将修改应用于它。同时,immer 还会通过比较草稿和基础版本来判断哪些属性已经发生了变化。这种技术被称为结构共享,可以大大提高不可变数据的创建速度。

produce 函数中,immer 使用 JavaScript 的 Proxy 对象来拦截草稿的读取和修改操作。这使得 immer 可以知道哪些属性已经被访问过,从而对未来的修改进行优化。同时,immer 还会将修改操作记录在一个操作队列中,待到最后完成所有操作时再将其应用于基础版本,以确保操作的顺序正确无误。

结语

immer.js 提供了一种简单而直观的方式来创建不可变数据类型,它能够极大地提高前端开发的效率和质量,并减少错误和副作用的出现。我们希望本文可以帮助你更好地理解 immer.js 并开始使用它。

来源:JavaScript中文网 ,转载请注明来源 https://www.javascriptcn.com/post/70827

纠错
反馈