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

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

相关推荐
神の愛1 小时前
左连接查询数据 left join
java·服务器·前端
小码哥_常3 小时前
解锁Android嵌入式照片选择器,让你的App体验丝滑起飞
前端
郑寿昌3 小时前
IIoT本体迁移的领域扩展机制
服务器·前端·microsoft
深海鱼在掘金4 小时前
Next.js从入门到实战保姆级教程(第十一章):错误处理与加载状态
前端·typescript·next.js
深海鱼在掘金4 小时前
Next.js从入门到实战保姆级教程(第十二章):认证鉴权与中间件
前端·typescript·next.js
energy_DT4 小时前
2026年十五五油气田智能增产装备数字孪生,CIMPro孪大师赋能“流动增产工厂”三维可视化管控
前端
龙猫里的小梅啊4 小时前
CSS(四)CSS文本属性
前端·css
MXN_小南学前端4 小时前
watch详解:与computed 对比以及 Vue2 / Vue3 区别
前端·javascript·vue.js
饭小猿人5 小时前
Flutter实现底部动画弹窗有两种方式
开发语言·前端·flutter
让学习成为一种生活方式5 小时前
pbtk v 3.5.0安装与使用--生信工具084
前端·chrome