【手写 Vuex 源码】第八篇 - Vuex 的 State 状态安装

news2024/12/26 12:14:16

一,前言

上一篇,主要介绍了 Vuex 模块安装的实现,针对 action、mutation、getter 的收集与处理,主要涉及以下几个点:

  • Vuex 模块安装的逻辑;
  • Vuex 代码优化;
  • Vuex 模块安装的实现;
  • Vuex 初始化流程梳理;

本篇,继续介绍 Vuex 模块相关概念:Vuex 的 State 状态安装;


二,前文梳理

  • 前面,通过依赖收集对 options 的格式化处理得到了“模块树”;
  • 又通过依赖安装对“模块树”进行递归操作:从根模块开始,将对应的 getter、mutation、action 统一放入 Store 类的 this._actions、this._mutations、this._wrappedGetters 中;

Vuex 的初始化流程如下:

  • 当项目引用并注册 vuex 插件时,即 Vuex.use(vuex),将执行 Vuex 插件中的 install 方法;
  • install 方法,接收外部传入的 Vue 实例,并通过 Vue.mixin 实现 store 实例的全局共享;
  • 项目中通过 new Vuex.Store(options) 配置 vuex 并完成 store 状态实例的初始化;
  • 在 Store 实例化阶段时,将会对 options 选项进行处理,此时完成 Vuex 模块收集和安装操作;
  • new Vue 初始化时,将 store 实例注入到 vue 根实例中(此时的 store 实例已实现全局共享);

接下来,继续在 installModule 方法中处理“模块树”的 State 状态;


三,Vuex 模块安装-State状态安装

1,State 状态的安装逻辑

对“模块树”中状态的安装,就是将所有子模块上的 State 状态,挂载到对应父模块的 State 状态上;

  • 处理范围:子模块,即 path.length > 0时,执行状态安装逻辑;
  • 处理逻辑:将子模块的状态 module.state,挂载到其父模块的状态上;

2,引出问题

所以,首先需要解决以下两个问题:

  • 问题 1:如何找到当前子模块对应的父模块?
  • 问题 2:如何将子模块状态挂载到父模块的状态上?

3,问题 1:如何找到当前子模块对应的父模块?

从“模块树”中,查找一个子模块对应的父模块,这个逻辑其实在模块收集时就已经写过了:

/**
 * 安装模块
 * @param {*} store       容器
 * @param {*} rootState   根状态
 * @param {*} path        所有路径
 * @param {*} module      格式化后的模块对象
 */
const installModule = (store, rootState, path, module) => {
  // 处理子模块:将子模块上的状态,添加到对应父模块的状态中;
  if(path.length > 0){
    // 从根状态开始逐层差找,找到当前子模块对应的父模块状态
    let parent = path.slice(0, -1).reduce((memo, current)=>{
        return memo[current]
    }, rootState)
  }
}

接下来,只需要向这个 parent 父模块的 State 状态中,添加当前子模块状态即可;

4,问题 2:如何将子模块状态挂载到父模块的状态上?

  • 子模块状态:module.state;
  • 父模块状态:parent;

那么,直接向父模块状态中添加子模块状态就可以了吗?

parent[path[path.length-1]] = module.state;

右侧的进度条告诉我们不会这么简单的:

  • 在 Vuex 中,模块是可以动态进行添加的;
  • 在 Vuex 中,状态应该是响应式的;

因此,我们希望动态添加的模块也是响应式的数据;

如果,直接向对象中添加一个不存在的属性,是无法被声明为响应式数据的;

所以,需要通过 Vue.set API 向父模块状态中添加子模块状态,以此实现对象新增属性为响应式数据;

这样,当 Vuex 动态注册模块时,新添加的状态属性就是响应式数据了;

备注:

  • 如果使用 Vue.set 向一个非响应式对象添加属性,相当于直接为普通对象添加属性并赋值;
  • 此时,即 resetStoreVM 方法执行前,parent就是一个普通对象,当 resetStoreVM 方法执行完成后,才是响应式数据,因此,使用 Vue.set在两种状态下都是兼容的;

5,代码实现

从根模块开始递归处理,将当前子模块状态定义到其对应父模块状态上:

  • 从根模块的状态开始找,返回当前模块所属的父模块 parent;
  • 将当前模块的 State 状态设置到父模块 parent 的 path[path.length-1] 属性中;
    即:将所有状态都设置到 rootState 上:
// src/vuex/store.js#installModule

const installModule = (store, rootState, path, module) => {
  // 处理子模块:将子模块上的状态,添加到对应父模块的状态中;
  if(path.length > 0){
    // 从根状态开始逐层差找,找到当前子模块对应的父模块状态
    let parent = path.slice(0, -1).reduce((memo, current)=>{
        return memo[current]
    }, rootState)
    // 支持 Vuex 动态添加模块,将新增状态直接定义成为响应式数据;
    Vue.set(parent, path[path.length-1], module.state);
  }
}

6,执行情况分析

  • 首次进入 installModule 方法,由于 path = [],不会进入状态安装逻辑;此时,会遍历根模块中的 actionsmutationsgetters 分别放到 store 实例中的 _actions_mutations_wrappedGetters 中;最后,通过当前模块 module.forEachChild 递归遍历子模块(深度优先递归),递归的终止条件是当父模块下不存在子模块时;

    备注:在遍历处理当前模块下的子模块时,完成 path 路径的拼接操作;

  • 非首次进入 installModule 方法,此时 path.length > 0, 进行子模块状态安装:先通过 path 找到当前子模块对应的父模块状态对象,并通过Vue.set向其中添加子模块状态,该属性被直接定义为响应式数据;

打印状态安装完成后的state,此时 state 包含 Vuex 中全部模块的状态:

// src/vuex/store.js

export class Store {
  constructor(options) {
    const state = options.state;

    // 收集所有模块中的action、mutation、getter 放到 Store 上
    this._actions = {};
    this._mutations = {};
    this._wrappedGetters = {};

    // 1,模块收集:options 格式化 -> Vuex 模块树
    this._modules = new ModuleCollection(options);
    console.log("格式化后的模块树对象", this._modules)

    // 2,模块安装:
    installModule(this, state, [], this._modules.root);
    console.log("模块安装结果:state", state)
  }
}

image.png

如上图所示:根模块状态中包含模块 A 和模块 B 的状态,模块 A 中包含模块 C 的状态;


四,结尾

本篇,主要介绍了 Vuex 的 State 状态安装,主要涉及以下几个点:

  • State 状态的安装逻辑;
  • 两个核心问题的思路;
  • 代码实现以及执行情况分析;

下一篇,继续介绍 Vuex 模块相关概念:Vuex 数据响应式的实现


维护日志

  • 20211008
    • 全篇重构,调整文章目录结构,添加核心逻辑分析;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/340378.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

leetcode-每日一题-2335(简单,贪心)

自己打表看一下过程就可以发现,其实就是每次选两个大的进行--之后秒数加1即可现有一台饮水机,可以制备冷水、温水和热水。每秒钟,可以装满 2 杯 不同 类型的水或者 1 杯任意类型的水。给你一个下标从 0 开始、长度为 3 的整数数组 amount &am…

元学习(Meta Learning)最全论文、视频、书籍资源整理

Meta Learning,叫做元学习或者 Learning to Learn 学会学习,包括Zero-Shot/One-Shot/Few-Shot 学习,模型无关元学习(Model Agnostic Meta Learning)和元强化学习(Meta Reinforcement Learning)。元学习是人工智能领域&…

Redis基础篇

Redis基础篇 来自黑马的redis课程的笔记 【黑马程序员Redis入门到实战教程,深度透析redis底层原理redis分布式锁企业解决方案黑马点评实战项目】 目录Redis基础篇一、初识Redis1.安装2. 数据结构二、Redis常见命令1. Redis通用命令2. string类型2.1 key的层级格式…

【成为架构师课程系列】预备架构 Pre-Architecture 的故事

目录 前言 Pre-architecture的故事 Pre-architecture 核心“四步法” 需求结构化 架构约束

1行Python代码去除图片水印,网友:一干二净

大家好,这里是程序员晚枫。 最近小明在开淘宝店(店名:爱吃火锅的少女),需要给自己的原创图片加水印,于是我上次给她开发了增加水印的功能:图片加水印,保护原创图片,一行…

伏并网低电压穿越技术

国内光伏并网低电压穿越要求 略: 低电压穿越方法 当前,光伏电站实现低电压穿越可通过两种方式,即增加硬件设备或者改变控制策略。本节对基于储能设备、基于无功补偿设备、基于无功电流电压支撑控制策略三种实现LVRT的典型方法进行介绍。 …

【MT7628】开发环境搭建-Fedora12一步一步设置共享文件夹

1.按照下图操作,打开设置界面 2.点击设置,弹出如下对话框

【博客620】prometheus如何优化远程读写的性能

prometheus如何优化远程读写的性能 场景 为了解决prometheus本地存储带来的单点问题,我们一般在高可用监控架构中会使用远程存储,并通过配置prometheus的remote_write和remote_read来对接 远程写优化:remote_write 远程写的原理&#xff1a…

Springboot扩展点之@PostConstruct

前言postContruct全限定类名是javax.annotation.PostConstruct,可以看出来其本身不是Spring定义的注解,但是Spring提供了具体的实现,所以这篇文章主要分析的是PostConstruct在Spring项目开发中的功能特性、实现方式和基本工作原理。功能特性从…

Linux c编程之Wireshark

Wireshark是一个网络报文分析软件,是网络应用问题分析必不可少的工具软件。网络管理员可以使用wireshark排查网络问题。程序开发人员可以用来分析应用协议、定位分析应用问题。无论是网络应用程序开发人员、测试人员、部署人员、技术支持人员,掌握wireshark的使用对于分析网络…

【LeetCode】每日一题(2)

目录 题目:1138. 字母板上的路径 - 力扣(Leetcode) 题目的接口: 解题思路: 代码: 过啦!!! 写在最后: 题目:1138. 字母板上的路径 - 力扣&am…

Python-项目实战--贪吃蛇小游戏-游戏框架搭建(2)

1.游戏框架搭建介绍pygame开发图像界面游戏的几个要素,并且把贪吃蛇游戏的整体框架搭建完成本节知识点包括:pygame的初始化和退出游戏主窗口游戏循环和游戏时钟主窗口背景颜色绘制文本pygame的坐标系游戏事件监听绘制图形定时器事件1.1pygame的初始化和退…

RocketMQ5.0.0消息存储<三>_消息转发与恢复机制

目录 一、消息转发 1. ReputMessageService线程初始化 2. 消息转发更新ConsumeQueue 3. 消息转发更新IndexFile 二、恢复机制 1. Broker加载存储文件 2. Broker正常退出的文件恢复 3. Broker异常退出的文件恢复 三、参考资料 一、消息转发 消息消费队列文件、索引文件…

七大设计原则之单一职责原则应用

目录1 单一职责原则介绍2 单一职责原则应用1 单一职责原则介绍 单一职责(Simple Responsibility Pinciple,SRP)是指不要存在多于一个导致类变更的原因。假设我们有一个 Class 负责两个职责,一旦发生需求变更,修改其中…

线程池的简单实现:Java线程池初学者必读指南

"作为一名Java开发者,是否曾经遇到过多线程并发的问题?线程数量过多时,会导致资源浪费,应用性能下降,甚至发生线程死锁的情况。那么,有没有一种方法可以有效地管理线程,避免这些问题呢&…

Matlab傅里叶谱方法求解一维波动方程

傅里叶谱方法求解基本偏微分方程—一维波动方程 一维波动方程 对于一根两端固定、没有受到任何外力的弦, 若只研究其中的一段, 在不太长的时间 里, 固定端来不及对这段弦产生影响, 则可以认为固定端是不存在的, 弦的长度为无限大。 这种无界 (−∞<x<∞)(-\infty<x&…

震源机制(Focal Mechanisms)之沙滩球(Bench Ball)

沙滩球包含如下信息&#xff1a; a - 判断断层类型&#xff0c;可根据球的颜色快速判断 b - 判断断层的走向(strike)&#xff0c;倾角(dip) c - 确定滑移角/滑动角(rake) 走向 &#xff0c;倾角&#xff0c;滑移角 如不了解断层的定义&#xff0c;可以先阅读&#xff1a;震…

windows下qt设置网卡ip信息+简单案列(图形化界面设置网卡IP)。

windows设置网卡ip信息的方法 文章目录windows设置网卡ip信息的方法前言一、QProcess修改网卡ip信息1.1 代码实例二、system修改网卡ip信息2.1 代码实例三、qt修改网卡信息案例3.1 设计方法3.2 代码实例3.3 功能测试前言 方法1&#xff1a;QProcess修改网卡ip信息&#xff1b;…

四种方式的MySQL安装

mysql安装常见的方法有四种序号 安装方式 说明1 yum\rpm简单、快速&#xff0c;不能定制参数2二进制 解压&#xff0c;简单配置就可使用 免安装 mysql-a.b.c-linux2.x-x86_64.tar.gz3源码编译 可以定制参数&#xff0c;安装时间长 mysql-a.b.c.tar.gz4源码制成rpm包 把源码制…

Spring boot 实战指南(四):登录认证(OAuth、Cookie、Session、Token)、Spring Security

文章目录一、登录认证方式1.OAuth 认证颁发令牌的四种方式2.Cookie/Session 认证(1)Cookie(2)Session3.Token认证基于JWT的Token认证(spring security)二、Spring boot整合Spring Security(前后端分离)1.快速入门2.认证3.授权参考&#xff1a; 教程 登录认证简介 OAuth 2.0 的一…