vuejs3.0 从入门到精通——Vuex 4.x —— state
Vuex 4.x —— state
https://vuex.vuejs.org/zh/guide/state.html
一、单一状态树
https://vuex.vuejs.org/zh/guide/state.html#单一状态树
Vuex 使用单一状态树——是的,用一个对象就包含了全部的应用层级状态。至此它便作为一个“唯一数据源 (SSOT)”而存在。这也意味着,每个应用将仅仅包含一个 store 实例。单一状态树让我们能够直接地定位任一特定的状态片段,在调试的过程中也能轻易地取得整个当前应用状态的快照。
单状态树和模块化并不冲突——在后面的章节里我们会讨论如何将状态和状态变更事件分布到各个子模块中。
存储在 Vuex 中的数据和 Vue 实例中的data
遵循相同的规则,例如状态对象必须是纯粹 (plain) 的。
以下是个人理解:
在Vuex中,我们使用单一状态树来管理应用的状态。这意味着所有的状态都被存储在一个单一的对象中,这个对象就是我们的“唯一数据源(SSOT)”。这种方式的优点在于,我们可以非常方便地跟踪和管理状态,同时在调试过程中,我们也可以轻松地获取到应用当前状态的完整快照。
单一状态树并不意味着我们不能将状态进行模块化管理。实际上,我们可以将状态以及状态的变更事件分布到各个子模块中。这样既可以保持状态的集中管理,又能够实现代码的模块化和可维护性。
同时,Vuex中的数据和Vue实例中的data遵循相同的规则,都是使用纯粹(plain)的对象。这意味着它们都是普通的JavaScript对象,不能包含任何特殊的Vue对象或函数。这样做的原因是为了保证状态的透明性和可预测性。只有纯粹的对象才能被序列化和反序列化,从而使得状态的变化可以被记录和跟踪。
二、在 Vue 组件中获得 Vuex 状态
https://vuex.vuejs.org/zh/guide/state.html#在 Vue 组件中获得 Vuex 状态
那么我们如何在 Vue 组件中展示状态呢?由于 Vuex 的状态存储是响应式的,从 store 实例中读取状态最简单的方法就是在计算属性中返回某个状态:
// 创建一个 Counter 组件
const Counter = {
template: `<div>{{ count }}</div>`,
computed: {
count () {
return store.state.count
}
}
}
每当store.state.count
变化的时候, 都会重新求取计算属性,并且触发更新相关联的 DOM。
然而,这种模式导致组件依赖全局状态单例。在模块化的构建系统中,在每个需要使用state的组件中需要频繁地导入,并且在测试组件时需要模拟状态。
以下是个人理解:
全局状态单例依赖:组件直接依赖于全局的 Vuex store,使得组件与全局状态紧密耦合。这违反了模块化和可重用性的原则,因为组件的行为受到了全局状态的影响,不利于代码维护和重用。
测试困难:由于组件依赖于全局状态,对其进行单元测试会变得困难。你需要模拟全局状态,以确保组件在测试环境下的行为符合预期。
状态管理不清晰:随着应用的复杂度增加,状态的来源和使用可能会变得不清晰。如果多个组件都依赖于全局状态,那么状态的改变可能会在不明显的地方引起副作用,增加调试的难度。
Vuex 通过 Vue 的插件系统将 store 实例从根组件中“注入”到所有的子组件里。且子组件能通过this.$store
访问到。让我们更新下Counter
的实现:
const Counter = { template: `<div>{{ count }}</div>`, computed: { count () { return this.$store.state.count } } }
三、mapState 辅助函数
当一个组件需要获取多个状态的时候,将这些状态都声明为计算属性会有些重复和冗余。为了解决这个问题,我们可以使用mapState
辅助函数帮助我们生成计算属性,让你少按几次键:
// 在单独构建的版本中辅助函数为 Vuex.mapState
import { mapState } from 'vuex'
export default {
// ...
computed: mapState({
// 箭头函数可使代码更简练
count: state => state.count,
// 传字符串参数 'count' 等同于 `state => state.count`
countAlias: 'count',
// 为了能够使用 `this` 获取局部状态,必须使用常规函数
countPlusLocalState (state) {
return state.count + this.localCount
}
})
}
当映射的计算属性的名称与 state 的子节点名称相同时,我们也可以给mapState
传一个字符串数组。
computed: mapState([ // 映射 this.count 为 store.state.count 'count' ])
四、对象展开运算符
mapState
函数返回的是一个对象。我们如何将它与局部计算属性混合使用呢?通常,我们需要使用一个工具函数将多个对象合并为一个,以使我们可以将最终对象传给computed
属性。但是自从有了对象展开运算符,我们可以极大地简化写法:
computed: { localComputed () { /* ... */ }, // 使用对象展开运算符将此对象混入到外部对象中 ...mapState({ // ... }) }
五、组件仍然保有局部状态
使用 Vuex 并不意味着你需要将所有的状态放入 Vuex。虽然将所有的状态放到 Vuex 会使状态变化更显式和易调试,但也会使代码变得冗长和不直观。如果有些状态严格属于单个组件,最好还是作为组件的局部状态。你应该根据你的应用开发需要进行权衡和确定。