immer原理解析

一. immer介绍

immer是一个高性能更新数据状态的工具库。对于状态有更新部分会创建新的副本对象,然后基于副本对象做数据更新,对于状态没有更新部分则是直接复用。

javascript 复制代码
import { produce } from 'immer'

const state = {
  name: 'zhansgan',
  age: 10,
}

const nextState = produce(state, draft => {
  draft.name = 'lisi'
})

二. immer原理

核心原理是使用Proxy对象,监听getset事件。

当触发get事件回调时,会创建key对应valueproxy对象。

当触发set事件回调时,会创建当前state副本对象,然后更新副本对象的属性值。

2.1 定义produce

  • 创建state对应的proxy对象
  • 调用recipe方法,执行更新逻辑
  • 调用finalize方法,返回state副本
javascript 复制代码
export function produce(base, recipe) {
  draft = createProxy(base)
  recipe(draft)
  return finalize(draft)
}

2.2 定义createProxy

通过Proxy.revocable方法获取proxy对象。该方法具体用法参考文档

javascript 复制代码
// 创建state副本
function prepareCopy(state) {
  if (!state.copy_) {
    state.copy_ = shallowCopy(state.base_)
  }
}

const traps = {
  get(state, prop) {
    if (prop === DRAFT_STATE) return state
    const source = latest(state)
    const value = source[prop]
    // 如果非对象或者数组类型,直接返回value
    if (!isDraftable(value)) return value
    const current = state.base_[prop]
    if (current === value) {
      prepareCopy(state)
      return (state.copy_[prop] = createProxy(value))
    }
    return value
  },
  set(state, prop, value) {
    if (!state.modified_) {
      state.modified_ = true
      prepareCopy(state)
    }
    const source = latest(state)
    const current = source[prop]
    if (Object.is(current, value)) return true
    state.copy_[prop] = value
    return true
  },
}

function createProxy(base) {
  let state = {
    base_: base, // 原始state
    copy_: null, // state副本
    draft_: null, // state proxy
    revoke_: null,
    modified_: false, // 记录当前state会否有修改过
  }
  const { proxy, revoke } = Proxy.revocable(state, traps)
  state.draft_ = proxy
  state.revoke_ = revoke
  return proxy
}

2.3 定义finalize

返回state副本对象,需要注意如果副本对象有属性值是proxy对象,需要递归处理。

javascript 复制代码
function finalize(draft) {
  const state = draft[DRAFT_STATE]
  if (!state.modified_) {
    state.revoke_()
    return state.base_
  }
  const result = state.copy_
  each(result, (key, value) => {
    if (isDraft(value)) result[key] = finalize(value)
  })
  state.revoke_()
  return result
}

三. immer使用准则

3.1 不要使用draft对象进行查询操作

当有查询诉求时,应当基于原始对象进行查询,而不该使用draft对象,因为会触发get事件回调,会创建新的proxy对象,会造成更多的性能开销。

javascript 复制代码
const remove = produce((list, element) => {  
  const index = list.indexOf(element) // this won't work!  
  const index = original(list).indexOf(element) // do this instead  
  if (index > -1) list.splice(index, 1)  
})

const values = [a, b, c]  
remove(values, a)

四. 总结

immer将原始state转换成proxy对象,当修改state数据时,可以通过setget事件细粒度创建修改数据部分的副本对象,基于副本对象做数据更新,而数据没有变更部分则直接复用,通过这种机制实现数据高效更新。代码仓库

创作不易,如果文章对你有帮助,那点个小小的赞吧,你的支持是我持续更新的动力!

相关推荐
Mr.Jessy9 分钟前
JavaScript高级:构造函数与原型
开发语言·前端·javascript·学习·ecmascript
白兰地空瓶2 小时前
🚀你以为你在写 React?其实你在“搭一套前端操作系统”
前端·react.js
爱上妖精的尾巴3 小时前
6-4 WPS JS宏 不重复随机取值应用
开发语言·前端·javascript
似水流年QC3 小时前
深入探索 WebHID:Web 标准下的硬件交互实现
前端·交互·webhid
陪我去看海3 小时前
测试 mcp
前端
speedoooo4 小时前
在现有App里嵌入一个AI协作者
前端·ui·小程序·前端框架·web app
全栈胖叔叔-瓜州4 小时前
关于llamasharp 大模型多轮对话,模型对话无法终止,或者输出角色标识User:,或者System等角色标识问题。
前端·人工智能
三七吃山漆4 小时前
攻防世界——wife_wife
前端·javascript·web安全·网络安全·ctf
用户47949283569154 小时前
面试官问"try-catch影响性能吗",我用数据打脸
前端·javascript·面试
GISer_Jing5 小时前
前端营销技术实战:数据+AI实战指南
前端·javascript·人工智能