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事件细粒度创建修改数据部分的副本对象,基于副本对象做数据更新,而数据没有变更部分则直接复用,通过这种机制实现数据高效更新。代码仓库

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

相关推荐
juruiyuan11110 分钟前
FFmpeg3.4 libavcodec协议框架增加新的decode协议
前端
Peter 谭31 分钟前
React Hooks 实现原理深度解析:从基础到源码级理解
前端·javascript·react.js·前端框架·ecmascript
周胡杰1 小时前
鸿蒙接入flutter环境变量配置windows-命令行或者手动配置-到项目的创建-运行demo项目
javascript·windows·flutter·华为·harmonyos·鸿蒙·鸿蒙系统
LuckyLay2 小时前
React百日学习计划——Deepseek版
前端·学习·react.js
gxn_mmf2 小时前
典籍知识问答重新生成和消息修改Bug修改
前端·bug
hj10432 小时前
【fastadmin开发实战】在前端页面中使用bootstraptable以及表格中实现文件上传
前端
乌夷2 小时前
axios结合AbortController取消文件上传
开发语言·前端·javascript
晓晓莺歌3 小时前
图片的require问题
前端
码农黛兮_463 小时前
CSS3 基础知识、原理及与CSS的区别
前端·css·css3
水银嘻嘻3 小时前
web 自动化之 Unittest 四大组件
运维·前端·自动化