最简实现vuex系列2- dispatch / getters

前言

昨天写了 commit/state,今天来看看 dispatch / getter

getter

有时候我们需要从 state 中得出派生状态,类似于 vue 中的 computed 属性,其实 vuex 也是利用了 computed

比如下面的这个例子:

ts 复制代码
createStore ({
 state:{
   age:10
 },
 getters:{
  calAge(state){
   //通过 state 的值 获取新的状态
   return state.age + 10
  }
 }
})

使用 _wrappedGetters 包装 getters

ts 复制代码
class Store<T> {
  constructor(options) {
  // getter 集合
   this._wrappedGetters = Object.create(null)
   const store = this
   
    Object.entries(options.getters).forEach(([key, fnValue]) => {
      this.registerGetter(store, key, fnValue)
    })
    
  }
  
   registerGetter(store: Store<T>, key, handler) {
    store._wrappedGetters[key] = function wrappedGetter(store) {
    // 不仅可以传递 state,也可以传递 getter
      return handler(store.state, store.getters)
    }
  }
}

store.util.js

最后 _wrappedGetters 形如

js 复制代码
{ 
  calAge:    
    function wrappedGetter(store){
        return [calAgefn](store.state,store.getters) 
    }
}

然后执行

ts 复制代码
function partial(fn, arg) {
 return function () {
   return fn(arg)
 }
}

class Store<T> {
 constructor(options) {
 
  Object.entries(options.getters).forEach(([key, fnValue]) => {
     this.registerGetter(store, key, fnValue)
   })

   this.resetStoreState(store)
 }
 
  resetStoreState(store){
   store.getters = {}
   const wrappedGetters = store._wrappedGetters
   const computedObj = {}
   const computedCache = {}

   Object.entries(wrappedGetters).forEach(([key,fn])=>{
     
     computedObj[key] = partial(fn, store)
     computedCache[key] = computed(() => computedObj[key]())

     Object.defineProperty(store.getters, key, {
       get: () => computedCache[key].value,
       enumerable: true
     })
   })
 }
}

使用 resetStoreState 进行缓存

这样做有几点好处

  • 计算函数fn被缓存在computedObj中,避免每次获取计算属性时都重新创建函数。
  • 计算属性被缓存在computedCache中,Vue框架会自动缓存计算属性的计算结果。
  • 通过computedObj和computedCache分离函数定义和属性读取,实现了计算函数的复用和结果缓存 最后通过 Object.defineProperty 定义 get 属性,实现了 getter的 可读

dispatch

dispatch 可以触发 在 actions 定义的方法,在 actions 中可以触发 commit/dispatch,也可以触发异步方法

ts 复制代码
class Store<T> { 
   constructor(options) {
    this._actions = Object.create(null)

   Object.entries(options.actions).forEach(([fnName, fnValue]) => {
      this.registerAction(store, fnName, fnValue)
    })
}


  registerAction(store, type, handler,){
    const entry = store._actions[type] || (store._actions[type] = []);
    entry.push(function wrappedActionHandler (payload) {
      let res = handler.call(store, {
        dispatch: store.dispatch,
        commit: store.commit,
        getters: store.getters,
        state: store.state,
      }, payload)

      if(!res || typeof res.then !== "function"){
        res = Promise.resolve(res)
      }
    })
  }
}

store-util.js

这个 dispatch 比较简单

ts 复制代码
dispatch(type, payload){
    const entry = this._actions[type]

    if (!entry) {
      return
    }

    const result = entry.length > 1
    ? Promise.all(entry.map(handler => handler(payload)))
    : entry[0](payload)

    return new Promise((resolve) => {
      result.then(res => {
        resolve(res)
      })
    })
  }

类型

ts 复制代码
type StoreOtpions<T> = {
  state?: T
  mutations?: Record<string, (s: T, payload: any) => void>,
  actions?: Record<string, (s: Store<T>,payload?:any) => void>,
  getters?: Record<string, (s: T, getters: StoreOtpions<T>["getters"]) => void>
}

ts 也能推导出 state 类型

总体来说,还是比较清晰的,也比较简单

相关推荐
qq. 280403398440 分钟前
js 原型链分析
开发语言·javascript·ecmascript
有趣的野鸭1 小时前
JAVA课程十一次实验课程主要知识点示例
java·前端·数据库
格鸰爱童话1 小时前
next.js(二)——从react到next.js
前端·javascript·react.js
Hammer Ray4 小时前
SourceMap知识点
javascript·sourcemap
西洼工作室4 小时前
项目环境变量配置全攻略
前端
阿珊和她的猫4 小时前
Webpack 优化:构建速度与包体积的双重提升
前端·webpack·node.js
阿珊和她的猫4 小时前
Webpack 打包体积优化:让应用更轻量、更高效
前端·webpack·状态模式
im_AMBER4 小时前
Vite + React 项目启动深度踩坑指南
前端·学习·react.js·前端框架
Hammer Ray4 小时前
前端开发基础概念(React)
前端·react.js·前端框架
Sunlightʊə7 小时前
2.登录页测试用例
运维·服务器·前端·功能测试·单元测试