复杂 Web 应用的状态管理思考

目录

  • 1.背景
  • 2.一个常见的多个组件共享状态的问题
  • 3.MVC 架构
  • 4.Flux 架构
  • 5.MVC 与 Flux 的对比
  • 6.Vue.js 的 Vuex 方案
  • 7.使用 Vuex 的正确姿势
  • 8.只有 Vuex 还不够...
  • 9.结语
  • 10.推荐阅读

1.背景

Web 页面开发的应用化趋势

随着 Web 能力的不断提升,用户的 Web 页面体验的要求,要求像原生一样,无刷新、离线缓存等和原生应用的功能。


Coding WebIDE

Web 开发已经进入组件化时代

Web页面开发组件化

2.一个常见的多个组件共享状态的问题

在下图中播放器组件上点赞以后,如何同步更新其他地方有关这个视频的点赞信息,比如右侧信息区组件、详情页?

点赞以后如何更新点赞的状态

可能的解决方案

  • 方案 1:刷新页面
  • 方案 2:把所有涉及到的接口都重新请求一遍
  • 方案 3: 直接更改 Dom
    ...

这些方案的问题

  • 方案 1: 体验差,会丢失用户的当前操作状态
  • 方案 2: 多余的请求,网络请求需要耗时
  • 方案 3:很难维护,也容易遗漏

问题的根源:
不同视图上的相同数据没有一个统一的来源,如果点赞信息能统一存储,只需要更新同一个地方,并且都从这个地方获取信息显示到视图上,就不会有这个问题了

3.MVC 架构

MVC 模式(Model–view–controller)是软件工程中的一种软件架构模式,把软件系统分为三个基本部分:模型(Model)、视图(View)和控制器(Controller)。

MVC组件之间的典型合作

比如点赞信息作为一个 Model, 每个视图都可以读取这个 Model,然后视图通过通知 Controller 去更新 Model,视图通过监听 Model 的变化再进行同步。

mvc的数据流动

4.Flux 架构

Flux 是由 Facebook 提出的,用于组织应用的一种架构,它基于一个简单的原则:数据在应用中单向流动。这就是所谓的“单向数据流”
,简单的记法是把数据比作鲨鱼:鲨鱼只能向前游

Paste_Image.png
flux架构设计

Flux 试图通过强制单向数据流来解决这个复杂度。在这种架构当中,Views 查询 Stores(而不是 Models),并且用户交互将会触发 Actions,Actions 则会被提交到一个集中的 Dispatcher 当中。当 Actions 被派发之后,Stores 将会随之更新自己并且通知 Views 进行修改。这些 Store 当中的修改会进一步促使 Views 查询新的数据。

5.MVC 与 Flux 的对比

EventBus 数量

  • MVC: 一个 model 一个
  • Flux: 只有一个

视图同步数据方式

  • MVC:view 监听 model 的事件
  • Flux: model 的改变自动同步到视图

查询和写入数据的区别

  • MVC:Model 暴露给 View,View 有可能会更新 model,像 backbone 就可以在 View 里调用this.model.set()
  • Flux: View 从 Store 获取的数据是只读的, Stores 只能通过 Actions 被更新

数据改变的可感知性

  • MVC:由于有多个 EventBus,所以很难去定位改变了 model 的具体来源
  • Flux: 任何状态的变化都必须通过 action 触发,而 action 又必须通过 dispatcher 走,所以整个应用的每一次状态变化都会从同一个地方流过


    复杂的Flux

6.Vuex

Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。
借鉴了 FluxRedux、和 The Elm Architecture。与其他模式不同的是,Vuex 是专门为 Vue.js 设计的状态管理库,以利用 Vue.js 的细粒度数据响应机制来进行高效的状态更新。

多个组件共享状态的问题

由于 Flux 架构有多个 store,我们的应用遇到多个组件共享状态时,单向数据流的简洁性很容易被破坏:

  • 多个视图依赖于同一状态。
  • 来自不同视图的行为需要变更同一状态。

问题的解决

  • 对于问题一,传参的方法对于多层嵌套的组件将会非常繁琐,并且对于兄弟组件间的状态传递无能为力。
  • 对于问题二,我们经常会采用父子组件直接引用或者通过事件来变更和同步状态的多份拷贝。以上的这些模式非常脆弱,通常会导致无法维护的代码。

最终解决方案

因此,我们为什么不把组件的共享状态抽取出来,以一个全局单例模式管理呢?在这种模式下,我们的组件树构成了一个巨大的“视图”,不管在树的哪个位置,任何组件都能获取状态或者触发行为!

另外,通过定义和隔离状态管理中的各种概念并强制遵守一定的规则,我们的代码将会变得更结构化且易维护。

这就是 Vuex 背后的基本思想。

7.使用 Vuex 的正确姿势

背景:前端数据来源的多样性

不同端的数据来源多样性

  • 服务端接口
  • 前端用户的输入
  • 客户端信息(比如设备信息等)

同一端数据来源的多样性

如果服务端的接口不是基于 restful 设计,或者接口内容是一些聚合的接口,比如服务端设计一个获取用户信息接口,返回了用户的 vip 信息、关注信息、粉丝信息等,同时设计了分别的获取 vip 信息接口、关注信息、粉丝信息接口,这样同一个内容在服务端这一端会有不同的来源。

如果前端组件之间对同一个状态的读取和操作没有做到复用,假设一个评论列表组件和弹幕组件之间,都用到了用户的评论信息,如果没有做到统一读取和更新,就会导致同样的状态信息有两个数据来源,最终导致同步问题。

数据读取,保证读取单一状态来源

需要保证不同的 View 或者组件读取某个状态信息都是从同一个地方读取,所以,回归到 Vuex 设计的本意上来,我们应该把同一状态信息集中管理。

数据存储,从聚合到原子

思路

  • 服务端尽量提供原子化的接口,比如 restful 风格
  • 前端数据层设计成原子化,不存储聚合信息,所有的聚合信息都拆开存储

前端数据存储原子化设计

下面是一个组队加速需求的状态设计:

{
  users: {
    [uid]: {
      userid: String,
      nickName: String,
      avatar: String,
      // 存储team的索引
      teams: String[]
    }
  },
  teams: {
    [groupId]: {
      teamId: String,
      taskId: String,
      // 存储user的索引
      users: String[]
    }
  },
  tasks: {
    [taskID]: {
      taskID: String,
      name: String,
      gcid: String,
      status: String,
      suffix: String,
      teams: String[]
    }
  },
  increases: {
    [gcid]: {
      [userId]: {
        uploadAmount: String,
        increaseAmount: String
      }
    }
  }
}

数据结构设计原则

  • 避免嵌套数据,数据结构扁平化
  • 双向关系存储索引,适当的冗余来提升读取的效率

8.有了 Vuex 还需要什么?

数据校验设计

Vuex 对 state 的写入没有统一的校验机制,复杂的单页面应用应保证 state 内存储的数据结构和类型都是一致的,可以在 Vuex 这一层做一层类似组件的 props 验证那样的验证机制。

数据缓存设计

需要一套完整的缓存机制,包括缓存的读、写及过期清理逻辑。

  • 持久化存储
    • localStorage
  • 会话期的缓存
    • 内存
    • sessionStorage
  • 跨页面会话期存储
    • localStorage

数据同步设计

  • 独立同步
  • 合并同步
  • 聚合同步

数据同步频率控制设计

就像一窝蜂的人去排队看演出,队伍很乱,看门的老大爷每隔 1 秒,让进一个人,这个叫 throttle,如果来了这一窝蜂的人,老大爷一次演出只让进一个人,下次演出才让下一个人进,这个就叫 debounce

  • throttle:节流,控制单位时间内对特定内容的同步请求的最大次数
  • debounce:去抖,连续的对同一个内容的同步请求我们只需要执行一次
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容