作为 Vue 全家桶的一员,Vuex 的重要性不言而喻,不管是用来管理状态,还是封装 Controler 都很好用

不过在一些体量较小的项目中,为了几个简单的状态或者处理函数而引入 Vuex,就像是高射炮打蚊子,大材小用了

这时候就可以模拟 Vuex,自己写一个简单的 Store, 用来管理状态并实时更新数据

一、构造函数

模拟 Vuex 的结构,创建一个 Class

export default class Store {
constructor({ states, actions, mutations }) {
// 状态
this.states = states || {};
// 异步函数
this.actions = actions || {};
// 同步函数
this.mutations = mutations || {};
}
// 调用 mutations 中的同步函数
commit = (fun, params) => {};
// 调用 actions 中的异步函数
dispatch = (fun, params) => {};
// 更新 states 的状态
update = (key, value) => {};
}

然后实例化一个 Store

import Store from './store';

import states from './states';
import actions from './actions';
import mutations from './mutations'; const store = new Store({
states,
actions,
mutations,
}); export default store;

然后挂到 vue 的原型上,通过 vue.$store 的方式使用,一个高仿 vuex 的架子就搭好了

// 在 main.js 中引入 Store
import store from './store/index';
Vue.prototype.$store = store;

二、实现操作函数(commit、dispatch、update

在 Vuex 中,如果需要更新 state 中的状态,需要通过 commit 调用 mutations 中的方法

而 mutations 的方法都具备一个默认的参数 state,因此 commit 方法可以这么写:

// 向 mutations 中的传入固定参数 state
commit = (fun, params) => {
this.mutations[fun](this.states, params);
};

不过由于一些历史遗留问题,我习惯用 this.states 的方式获取 state(这个习惯不好),所以改成了这样:

  commit = (fun, params) => {
if (fun) {
this.mutations[fun].call(this, params);
} else {
return false;
}
};

类似的 actions 和 update 可以参考 commit 的写法

三、响应式对象

目前的 store 有一个致命的问题:state 更新之后并不会即时渲染到视图层

这时候 Vue 2.6.0 新增的 observable() 就派上用场了

如果将一个对象作为入参传给 Vue.observable() ,经过处理之后,这个对象在 Vue 内就可以实时更新

其返回值可以直接用于 render 和 computed 中,并且会在发生改变时触发相应的更新

于是 Store 的构造函数需要改一改:

  constructor({ states, actions, mutations }) {
// 状态
this.states = Vue.observable(states || {});
// 异步函数
this.actions = Vue.observable(actions || {});
// 同步函数
this.mutations = Vue.observable(mutations || {});
}

⚠️注意:

假如对象 obj 经过 observable() 处理之后,赋值给了新对象 new_obj

在 Vue 2.x 中,直接修改 obj 也会触发 new_obj 的更新

但在 Vue 3.x 中,由于响应机制的变更,只有修改 new_obj 才能触发视图层的更新

所以在使用 observable() 的时候,最好始终操作使用 observable() 处理后的 new_obj

四、简单用用

超低配的 Vuex 已经写好了,上面已经把 store 挂到 Vue 的原型上,所以可以直接使用

假如 state 中已经存在一个状态 name,在组件中可以通过 computed 去获取

computed: {
name() {
return this.$store.states.name;
},
}

如果需要修改状态,可以通过 $store.update()

methods: {
updateName(val) {
this.$store.update('name', val);
}
}

或者使用 $store.commit() 调用 mutations 中的方法

methods: {
commitName(val) {
this.$store.commit('handleNameChange', val);
}
}

大功告成~

结合 Vue.observable 写一个简易 Vuex的更多相关文章

  1. Summer——从头开始写一个简易的Spring框架

    Summer--从头开始写一个简易的Spring框架                ​ 参考Spring框架实现一个简易类似的Java框架.计划陆续实现IOC.AOP.以及数据访问模块和事务控制模块. ...

  2. 手写一个简易的IOC

    这个小项目是我读过一点Spring的源码后,模仿Spring的IOC写的一个简易的IOC,当然Spring的在天上,我写的在马里亚纳海沟,哈哈 感兴趣的小伙伴可以去我的github拉取代码看着玩 地址 ...

  3. 来,我们手写一个简易版的mock.js吧(模拟fetch && Ajax请求)

    预期的mock的使用方式 首先我们从使用的角度出发,思考编码过程 M1. 通过配置文件配置url和response M2. 自动检测环境为开发环境时启动Mock.js M3. mock代码能直接覆盖g ...

  4. 如何用 Python 写一个简易的抽奖程序

    不知道有多少人是被这个头图骗进来的:) 事情的起因是这样的,上周有同学问小编,看着小编的示例代码敲代码,感觉自己也会写了,如果不看的话,七七八八可能也写的出来,但是一旦自己独立写一段程序,感觉到无从下 ...

  5. 手写一个简易的多周期 MIPS CPU

    一点前言 多周期 CPU 相比单周期 CPU 以及流水线 CPU 实现来说其实写起来要麻烦那么一些,但是相对于流水线 CPU 和单周期 CPU 而言,多周期 CPU 除了能提升主频之外似乎并没有什么卵 ...

  6. 写一个简易web服务器、ASP.NET核心知识(4)

    前言 昨天尝试了,基于对http协议的探究,我们用控制台写了一个简单的浏览器.尽管浏览器很low,但是对于http协议有个更好的理解. 说了上面这一段,诸位猜到我要干嘛了吗?(其实不用猜哈,标题里都有 ...

  7. 写一个简易浏览器、ASP.NET核心知识(3)

    前言 先在文章前面说好了,省得大家发现我根本没有这样的头发,duang的一下一堆人骂我. 这篇文章的标题有点大,其实挺low的,我需要在开头解释一下.我这里只想写一个小的控制台,旨在模拟浏览器的htt ...

  8. Vue.js写一个SPA登录页面的过程

    技术栈 vue.js 主框架 vuex 状态管理 vue-router 路由管理 一般过程 在一般的登录过程中,一种前端方案是: 检查状态:进入页面时或者路由变化时检查是否有登录状态(保存在cooki ...

  9. 用python 10min手写一个简易的实时内存监控系统

    简易的内存监控系统 本文需要有一定的python和前端基础,如果没基础的,请关注我后续的基础教程系列博客 文章github源地址,还可以看到具体的代码,喜欢请在原链接右上角加个star 腾讯视频链接 ...

随机推荐

  1. Flask框架整理及配置文件

    阅读目录 Flask目录结构(蓝图) pro_flask包的init.py文件, 用于注册所有的蓝图 manage.py文件,作为整个项目的启动文件 views包中的blog.py,必须要通过sess ...

  2. 201871010108-高文利《面向对象程序设计(java)》第八周学习总结

    项目 内容 这个作业属于哪个课程 <任课教师博客主页链接> https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 <作业链接地址> ht ...

  3. redis常用命令手册大全

    一.五种数据类型1.Redis字符串StringString 是最简单的类型,你可以理解成与 Memcached 是一模一样的类型,一个 key 对应一个value,其上支持的操作与 Memcache ...

  4. 莫烦TensorFlow_02 Session的两种方法

    import tensorflow as tf matrix1 = tf.constant([[3,3]]) # 1X2 matrix2 = tf.constant([[2], [2]]) produ ...

  5. python27期尚哥讲TFTP:

    TFTP介绍 :TFTP(Trivial File Transfer Protocol,简单⽂件传输协议)是TCP/IP协议簇中的⼀个⽤来在客户端与服务器之间进⾏简单⽂件传输的协议使用tftp这个协议 ...

  6. 【使用篇二】SpringBoot文件上传(5)

    一.单个文件上传 1. 在static目录下创建upload.html <!DOCTYPE html> <html> <head> <meta charset ...

  7. 【oracle】substr函数 字符截取

  8. Spring Cloud微服务安全实战_4-4_OAuth2协议与微服务安全

    接上篇文章,在这个流程中,PostMan可以代表客户端应用,订单服务是资源服务器,唯一缺少的是 认证服务器 ,下面来搭建认证服务器 项目结构: Pom.xml : DependencyManager ...

  9. Spring 中的异常处理

    工作中遇到这样的同事,问他活干完吗,他说开发好了,结果测试时发现各种异常情况未处理,联调测试时各种未知错误,最后联调完成比他预期的两倍工作量还多.这样的开发如果是新人还可以原谅,如果有工作经验且出现多 ...

  10. 初探Java设计模式5:一文了解Spring涉及到的9种设计模式

    本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutorial 喜欢的话麻烦点下 ...