一、Vuex概述

引用官方文档的话解释什么是Vuex

Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式 + 库。它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证状态以一种可预测的方式发生变化。
设想每个Vue组件都只需要关心自己的数据和数据处理逻辑,组件之间完全独立,没有共享数据的需求,那么web应用会非常简单。 但是实际情况是,一个web应用中不同组件常常会有数据共享的需求。例如文档类型的应用,编辑的内容和大纲之间要共享数据,大纲列表组件需要根据编辑的内容生成文档大纲、电商类型的应用,商品展示列表需要根据用户选择的分类展示相应的商品,那么商品列表组件需要知道分类选项组件中用户选择的是哪个类别。 我们知道Vue父组件可以和子组件之间通过props、事件、ref引用来进行通信,但是这种组件之间的通信方式在规模较大的应用中会有些力不从心。因此我们更倾向于将不同组件通用的数据提取出来统一管理。 我们知道Vue组件根据数据渲染视图,不同的数据对应不同的视图。我们也可以理解为web应用处于不同的“状态”,每个状态对应数据的一组取值。因此我们将数据抽出来统一管理也可以称为“状态管理”。 总之,Vuex就是专为Vue.js开发的状态管理工具,用于解决组件之间数据共享的需求。 状态管理工具都需要包含哪些要素?
  1. 初始状态
  2. 改变状态
  3. 监听状态的改变

首先我们看下Vuex都有哪些API,然后说明这些API的作用。

  1. state
  2. getters
  3. mutations
  4. actions
  5. module
  6. 辅助函数:mapState、mapGetters、mapMutations、mapActions
  7. createStore

其中state和getters用来保存状态;mutations和actions用来改变状态;监听状态用的是Vue组件中的computed属性;module是用来组织整个应用的状态管理代码,使状态划分模块,更易于管理;辅助函数用来在监听状态时候简化代码,createStore则用来创建状态管理对象。

  1. Vuex状态管理就是创建一个对象(store),这个对象上面保存了应用中大部分数据,组件可以通过store获取数据,也可以改变数据的值。上面说的这些能力一个普通的js对象也可以做到。store和普通的对象的区别是:
  1. Vuex 的状态存储是响应式的。当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新。
  2. 你不能直接改变 store 中的状态。改变 store 中的状态的唯一途径就是显式地提交 (commit) mutation。这样使得我们可以方便地跟踪每一个状态的变化,从而让我们能够实现一些工具帮助我们更好地了解我们的应用

Vuex的数据流是组件中触发Action,Action提交Mutations,Mutations修改State。 组件根据 States或Getters来渲染页面。

参考官网的图示Vuex官网

二、Vuex基本使用

注意这里使用Vuex版本为4


基本使用

首先我们来使用Vuex中的createStore方法创建一个store,我们注意到创建store的对象中目前有state和mutations两个属性。state中保存着我们需要使用的数据 count,mutations里面是改变state中数据的方法,方法接受一个state参数,通过改变这个state参数的属性(count)的值就可以修改状态了。

执行了store.commit(‘increase’);之后,Vuex会执行mutations中的increase方法,让count加一。


  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. count: 0
  5. },
  6. mutations: {
  7. increase(state) {
  8. state.count++;
  9. }
  10. }
  11. });
  12. store.commit('increase');
  13. console.log(store.state.count); // 1

mutations还可以接收参数

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. count: 0
  5. },
  6. mutations: {
  7. increase(state, {count}) {
  8. state.count = count;
  9. }
  10. }
  11. });
  12. store.commit('increase', {count: 2});
  13. console.log(store.state.count); // 2

getters

getters和state的关系类似于Vue组件data属性和computed属性的关系,getters根据state或者其他getters计算出另一个变量的值,当其依赖的数据变化时候,它也会实时更新。

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. count: 0
  5. },
  6. getters: {
  7. msg(state) {
  8. return state.count % 2 === 0 ? '偶数': '奇数';
  9. }
  10. },
  11. mutations: {
  12. increase(state) {
  13. state.count++;
  14. }
  15. }
  16. });
  17. store.commit('increase');
  18. console.log(store.state.count, store.getters.msg); // 1 "奇数"

actions


既然已经有了mutations可以改变state,为什么还需要actions呢?因为mutations不应该用于异步修改状态。实际上mutations是可以异步修改状态的,比如:

  1. mutations: {
  2. increase(state) {
  3. setTimeout(() => {
  4. state.count++;
  5. }, 1000);
  6. }
  7. }

但是这样做的话,Vuex是无法知道修改state.count的时机的,因为它是在异步回调里面指定的,因此Vuex无法在调试工具中打印出我们实际改变state的操作。

因此Vuex中有actions API,在actions中可以进行异步操作,在actions中可以提交mutations,也可以触发其他的action。

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. count: 0
  5. },
  6. mutations: {
  7. increase(state) {
  8. state.count++;
  9. }
  10. },
  11. actions: {
  12. increase(context) {
  13. setTimeout(() => {
  14. // contex.dispatch可以用于触发其他action
  15. context.commit('increase');
  16. }, 1000);
  17. }
  18. }
  19. });
  20. store.dispatch('increase');
  21. console.log(store.state.count); // 0
  22. setTimeout(() => {
  23. console.log(store.state.count); // 1
  24. }, 1000);

三、Vuex modules

通常在一个web应用中,会有很多数据,都放在一个store里面会让数据很混乱,因此我们应该根据功能模块将数据划分成一个一个的模块。

Vuex支持将store划分成模块,并且模块还可以嵌套。

下面看官方文档上的多个模块的示例:

在调用createStore时候传入的对象中,提供modules属性,传入两个子模块。

  1. import {createStore} from 'vuex';
  2. const moduleA = {
  3. state: {
  4. name: 'a'
  5. }
  6. };
  7. const moduleB = {
  8. state: {
  9. name: 'b'
  10. }
  11. };
  12. const store = createStore({
  13. modules: {
  14. a: moduleA,
  15. b: moduleB
  16. }
  17. });
  18. console.log(store.state.a.name); // a
  19. console.log(store.state.b.name); // b

看下包含嵌套子模块时候,访问子模块的getters和state的示例,

下面的示例中,有两个子模块a和b,其中a中还有嵌套的子模块c,

(注意默认情况下,所有子模块的getters、mutations和actions都是注册在全局的)

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. counter: 0
  5. },
  6. getters: {
  7. counter10times(state) {
  8. return state.counter * 10;
  9. }
  10. },
  11. modules: {
  12. a: {
  13. state: {aName: 'A·a'},
  14. aGetters: {
  15. aFirstName(state) {
  16. return state.aName.split('·')[0];
  17. }
  18. },
  19. modules: {
  20. c: {
  21. state: {cName: 'C·c'},
  22. getters: {
  23. cFirstName(state) {
  24. return state.cName.split('·')[0];
  25. }
  26. }
  27. }
  28. }
  29. },
  30. b: {
  31. state: {bName: 'B·b'},
  32. getters: {
  33. bFirstName(state) {
  34. return state.bName.split('·')[0];
  35. },
  36. bNewName(state, getters, rootState, rootGetters) {
  37. // 访问局部state
  38. const {bName} = state;
  39. // 访问全局state
  40. const {a: {c: {cName}}} = rootState;
  41. // 访问局部getters
  42. const {bFirstName} = getters;
  43. // 访问全局getters
  44. const {cFirstName} = rootGetters;
  45. return `${bName} ${bFirstName} ${cName} ${cFirstName}`;
  46. }
  47. }
  48. }
  49. }
  50. });
  51. // 子模块的state通过子模块路径访问
  52. console.log(store.state.a.c.cName);
  53. // 子模块的getters都注册到了全局,在store.getters下面直接能访问到
  54. console.log(store.getters.bNewName);

下面是一个多模块,commit mutation和dispatch action的示例,

模块可以提交其他模块的mutation,从而改变其他的模块的状态;模块也可以触发其他模块的action

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. counter: 0
  5. },
  6. mutations: {
  7. increaseCounter(state) {
  8. state.counter++;
  9. }
  10. },
  11. modules: {
  12. a: {
  13. state: {aName: 'A·a'},
  14. mutations: {
  15. changeAName(state) {
  16. state.aName = 'A-a';
  17. }
  18. },
  19. actions: {
  20. callChangeCNameAsync({dispatch}) {
  21. // 触发其他模块的action
  22. setTimeout(() => {
  23. dispatch('changeCNameAsync');
  24. }, 500);
  25. }
  26. },
  27. modules: {
  28. c: {
  29. state: {cName: 'C·c'},
  30. mutations: {
  31. changeCName(state, payload) {
  32. state.cName = `C-c-${payload.suffix}`;
  33. }
  34. },
  35. actions: {
  36. changeCNameAsync({commit, rootState}) {
  37. setTimeout(() => {
  38. // 提交其他模块的mutation,mutation是全局的
  39. commit('increaseCounter');
  40. // 提交局部模块的mutation
  41. commit('changeCName', {
  42. suffix: rootState.counter
  43. });
  44. }, 500);
  45. }
  46. }
  47. },
  48. }
  49. },
  50. b: {
  51. state: {bName: 'B·b'},
  52. mutations: {
  53. changeBName(state) {
  54. state.bName = 'B-b';
  55. }
  56. }
  57. }
  58. }
  59. });
  60. // 全局的commit
  61. store.commit('increaseCounter');
  62. console.log(store.state.counter); // 1
  63. // 子模块mutation注册到全局了
  64. store.commit('changeCName', {suffix: '123'});
  65. console.log(store.state.a.c.cName); // C-c-123
  66. // 子模块commit其他模块的mutation
  67. store.dispatch('changeCNameAsync');
  68. setTimeout(() => {
  69. console.log(store.state.a.c.cName); // C-c-2
  70. }, 1000);
  71. // 子模块dispatch其它模块的action
  72. store.dispatch('callChangeCNameAsync');
  73. setTimeout(() => {
  74. console.log(store.state.a.c.cName); // C-c-3
  75. }, 1500);

之前提到默认情况下,所有模块的getters、mutations和actions都是注册到全局的,这样如果多个子模块的getters、mutations和actions中有同名时候,会导致覆盖,引起问题。因此通常我们需要给子模块加命名空间。

给子模块加命名空间的方式是给子模块加namespaced属性并赋值为true。

加了命名空间后,访问state的方式不变(因为默认state也不是注册到全局的),访问getters时候需要加命名空间前缀,如果访问模块自身子模块的getters、提交mutations、触发actions时候,只需要加相对路径前缀,不需要加自身命名空间前缀,例如模块a访问其子模块c时候,不需要加’a/c’前缀,只需要’c’就可以了。

看下多模块(包含嵌套模块情况)时候访问state和getters的示例:

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. counter: 0
  5. },
  6. getters: {
  7. counter10times(state) {
  8. return state.counter * 10;
  9. }
  10. },
  11. modules: {
  12. a: {
  13. namespaced: true,
  14. state: {aName: 'A·a'},
  15. getters: {
  16. aFirstName(state) {
  17. return state.aName.split('·')[0];
  18. }
  19. },
  20. modules: {
  21. c: {
  22. namespaced: true,
  23. state: {cName: 'C·c'},
  24. getters: {
  25. cFirstName(state) {
  26. return state.cName.split('·')[0];
  27. }
  28. }
  29. }
  30. }
  31. },
  32. b: {
  33. namespaced: true,
  34. state: {bName: 'B·b'},
  35. getters: {
  36. bNewName(state, getters, rootState, rootGetters) {
  37. // 局部state
  38. const bName = state.bName.split('·')[0];
  39. // 其他模块的getter
  40. const cFirstName = rootGetters['a/c/cFirstName'];
  41. // 其他模块的state
  42. const aName = rootState.a.aName;
  43. return `${bName} ${cFirstName} ${aName}`;
  44. }
  45. }
  46. }
  47. }
  48. });
  49. // getters命名空间
  50. console.log(store.getters['b/bNewName']); // B C A·a
  51. // 子节点state仍然是通过节点路径访问
  52. console.log(store.state.a.c.cName); // C·c

看下在使用了命名空间的多模块的提交mutations和触发actions

  1. import {createStore} from 'vuex';
  2. const store = createStore({
  3. state: {
  4. counter: 0
  5. },
  6. mutations: {
  7. increaseCounter(state) {
  8. state.counter++;
  9. }
  10. },
  11. modules: {
  12. a: {
  13. namespaced: true,
  14. state: {aName: 'A·a'},
  15. mutations: {
  16. changeAName(state) {
  17. state.aName = 'A-a';
  18. }
  19. },
  20. actions: {
  21. callChangeCNameAsync({dispatch}) {
  22. // 触发子模块的action,是相对于自身的路径,不需要加a前缀
  23. setTimeout(() => {
  24. dispatch('c/changeCNameAsync');
  25. }, 500);
  26. }
  27. },
  28. modules: {
  29. c: {
  30. namespaced: true,
  31. state: {cName: 'C·c'},
  32. mutations: {
  33. changeCName(state, payload) {
  34. state.cName = `C-c-${payload.suffix}`;
  35. }
  36. },
  37. actions: {
  38. changeCNameAsync({commit, rootState}) {
  39. setTimeout(() => {
  40. // 提交其他模块的mutation,mutation是全局的
  41. commit('increaseCounter', null, {root: true});
  42. // 提交局部模块的mutation,不需要加前缀
  43. commit('changeCName', {
  44. suffix: rootState.counter
  45. });
  46. }, 500);
  47. }
  48. }
  49. },
  50. }
  51. },
  52. b: {
  53. namespaced: true,
  54. state: {bName: 'B·b'},
  55. mutations: {
  56. changeBName(state) {
  57. state.bName = 'B-b';
  58. }
  59. }
  60. }
  61. }
  62. });
  63. // 全局的commit
  64. // 注意加了命名空间之后,提交根模块的mutation和触发根模块的action时候,都需要加上{root: true}的选项
  65. store.commit('increaseCounter', null, {root: true});
  66. console.log(store.state.counter); // 1
  67. // 子模块mutation注册到全局了
  68. store.commit('a/c/changeCName', {suffix: '123'});
  69. console.log(store.state.a.c.cName); // C-c-123
  70. // 子模块commit其他模块的mutation
  71. store.dispatch('a/c/changeCNameAsync');
  72. setTimeout(() => {
  73. console.log(store.state.a.c.cName); // C-c-2
  74. }, 1000);
  75. // 子模块dispatch其它模块的action
  76. store.dispatch('a/callChangeCNameAsync');
  77. setTimeout(() => {
  78. console.log(store.state.a.c.cName); // C-c-3
  79. }, 1500);

四、在Vue组件中使用Vuex

1. 注入store

使用Vue3、Vuex4版本,通过如下方式向注入store,

  1. import { createApp } from 'vue';
  2. import App from './App.vue';
  3. import {createStore} from 'vuex';
  4. const store = createStore({
  5. state: {
  6. counter: 0
  7. },
  8. getters: {
  9. counter10times(state) {
  10. return state.counter * 10;
  11. }
  12. },
  13. mutations: {
  14. increaseCounter(state) {
  15. state.counter++;
  16. }
  17. },
  18. modules: {
  19. a: {
  20. namespaced: true,
  21. state: {aName: 'A·a'},
  22. getters: {
  23. aFirstName(state) {
  24. return state.aName.split('·')[0];
  25. }
  26. },
  27. mutations: {
  28. changeAName(state) {
  29. state.aName = 'A-a';
  30. }
  31. },
  32. actions: {
  33. callChangeCNameAsync({dispatch}) {
  34. // 触发子模块的action,相对与自身的路径
  35. setTimeout(() => {
  36. dispatch('c/changeCNameAsync');
  37. }, 500);
  38. }
  39. },
  40. modules: {
  41. c: {
  42. namespaced: true,
  43. state: {cName: 'C·c'},
  44. getters: {
  45. cFirstName(state) {
  46. return state.cName.split('·')[0];
  47. }
  48. },
  49. mutations: {
  50. changeCName(state, payload) {
  51. state.cName = `C-c-${payload.suffix}`;
  52. }
  53. },
  54. actions: {
  55. changeCNameAsync({commit, rootState}) {
  56. setTimeout(() => {
  57. // 提交其他模块的mutation,mutation是全局的
  58. commit('increaseCounter', null, {root: true});
  59. // 提交局部模块的mutation,不需要加前缀
  60. commit('changeCName', {
  61. suffix: rootState.counter
  62. });
  63. }, 500);
  64. }
  65. }
  66. }
  67. }
  68. },
  69. b: {
  70. namespaced: true,
  71. state: {bName: 'B·b'},
  72. getters: {
  73. bNewName(state, getters, rootState, rootGetters) {
  74. // 局部state
  75. const bName = state.bName.split('·')[0];
  76. // 其他模块的getter
  77. const cFirstName = rootGetters['a/c/cFirstName'];
  78. // 其他模块的state
  79. const aName = rootState.a.aName;
  80. return `${bName} ${cFirstName} ${aName}`;
  81. }
  82. },
  83. mutations: {
  84. changeBName(state) {
  85. state.bName = 'B-b';
  86. }
  87. }
  88. }
  89. }
  90. });
  91. createApp(App).use(store).mount('#app');

将刚才的加了命名空间的store注入到Vue组件树中。这样在所有的Vue组件中,都能够通过this.$store方式访问store。

Vue组件通过computed属性来监听store的数据变化。

看下面的示例,computed依赖了this.$store里面的一些模块的state和getters,并将计算结果展示在界面上。

  1. <template>
  2. <div>
  3. {{counter}}
  4. {{bName}}
  5. {{cFirstName}}
  6. </div>
  7. </template>
  8. <script>
  9. export default {
  10. computed: {
  11. counter() {
  12. return this.$store.state.counter;
  13. },
  14. bName() {
  15. return this.$store.state.b.bName;
  16. },
  17. cFirstName() {
  18. return this.$store.getters['a/c/cFirstName'];
  19. }
  20. }
  21. }
  22. </script>
  23. <style>
  24. #app {
  25. margin-top: 60px;
  26. }
  27. </style>

下面看下Vue组件中改变状态的示例,

可以看到Vue组件中通过methods方法调用this.$store的commit和dispatch方法来提交修改和触发action。

  1. /**
  2. * @file main.js
  3. */
  4. import { createApp } from 'vue';
  5. import App from './App.vue';
  6. import {createStore} from 'vuex';
  7. const store = createStore({
  8. state: {
  9. counter: 0
  10. },
  11. getters: {
  12. counter10times(state) {
  13. return state.counter * 10;
  14. }
  15. },
  16. mutations: {
  17. increaseCounter(state) {
  18. state.counter++;
  19. }
  20. },
  21. modules: {
  22. a: {
  23. namespaced: true,
  24. state: {aName: 'A·a'},
  25. getters: {
  26. aFirstName(state) {
  27. return state.aName.split('·')[0];
  28. }
  29. },
  30. mutations: {
  31. changeAName(state) {
  32. state.aName = 'A-a';
  33. }
  34. },
  35. actions: {
  36. callChangeCNameAsync({dispatch}) {
  37. // 触发子模块的action,相对于自身的路径
  38. setTimeout(() => {
  39. dispatch('c/changeCNameAsync');
  40. }, 500);
  41. }
  42. },
  43. modules: {
  44. c: {
  45. namespaced: true,
  46. state: {cName: 'C·c'},
  47. getters: {
  48. cFirstName(state) {
  49. return state.cName.split('·')[0];
  50. }
  51. },
  52. mutations: {
  53. changeCName(state, payload) {
  54. state.cName = `C-c-${payload.suffix}`;
  55. }
  56. },
  57. actions: {
  58. changeCNameAsync({commit, rootState}) {
  59. setTimeout(() => {
  60. // 提交其他模块的mutation,mutation是全局的
  61. commit('increaseCounter', null, {root: true});
  62. // 提交局部模块的mutation,不需要加前缀
  63. commit('changeCName', {
  64. suffix: rootState.counter
  65. });
  66. }, 500);
  67. }
  68. }
  69. }
  70. }
  71. },
  72. b: {
  73. namespaced: true,
  74. state: {bName: 'B·b'},
  75. getters: {
  76. bNewName(state, getters, rootState, rootGetters) {
  77. // 局部state
  78. const bName = state.bName.split('·')[0];
  79. // 其他模块的getter
  80. const cFirstName = rootGetters['a/c/cFirstName'];
  81. // 其他模块的state
  82. const aName = rootState.a.aName;
  83. return `${bName} ${cFirstName} ${aName}`;
  84. }
  85. },
  86. mutations: {
  87. changeBName(state) {
  88. state.bName = 'B-b';
  89. }
  90. }
  91. }
  92. }
  93. });
  94. createApp(App).use(store).mount('#app');
  1. /**
  2. * @file App.vue
  3. */
  4. <template>
  5. <div>
  6. {{counter}}
  7. {{bName}}
  8. {{cFirstName}}
  9. <button @click="modifyBName">修改bname</button>
  10. <button @click="modifyCName">修改cname</button>
  11. </div>
  12. </template>
  13. <script>
  14. export default {
  15. computed: {
  16. counter() {
  17. return this.$store.state.counter;
  18. },
  19. bName() {
  20. return this.$store.state.b.bName;
  21. },
  22. cFirstName() {
  23. return this.$store.getters['a/c/cFirstName'];
  24. }
  25. },
  26. methods: {
  27. modifyBName() {
  28. this.$store.commit('b/changeBName');
  29. },
  30. modifyCName() {
  31. this.$store.dispatch('a/callChangeCNameAsync');
  32. }
  33. }
  34. }
  35. </script>
  36. <style>
  37. #app {
  38. margin-top: 60px;
  39. }
  40. </style>

2. 辅助函数

我们知道我们使用Vuex时候,通过computed绑定store的state和getters的数据,通过methods中调用this.$store的commit和dispatch方法来改变状态。

但是每次都要写this.$store,当需要绑定的数据多的时候会比较繁杂,因此Vuex提供了辅助函数来简化代码。辅助函数包括

  1. mapState
  2. mapGetters
  3. mapMutations
  4. mapActions

其中mapState和mapGetters将映射到computed属性中,mapMutations和mapActions映射到methods属性中。

用法见下面示例

  1. <template>
  2. <div>
  3. {{counter}}
  4. {{bName}}
  5. {{counter10times}}
  6. {{cFirstName}}
  7. <button @click="modifyBName">修改bname</button>
  8. <button @click="modifyCName">修改cname</button>
  9. </div>
  10. </template>
  11. <script>
  12. import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'
  13. export default {
  14. computed: {
  15. ...mapState({
  16. // 将this.$store.state.counter映射为counter
  17. counter: state => state.counter,
  18. // 也可以这样实现
  19. // counter: 'counter',
  20. bName: state => state.b.bName
  21. }),
  22. // 全局的getters
  23. ...mapGetters(['counter10times']),
  24. // 也可以这样实现,指定组件中的数据名称
  25. // ...mapGetters({
  26. // counter10times: 'counter10times'
  27. // }),
  28. // 子模块的getters
  29. ...mapGetters({
  30. cFirstName: 'a/c/cFirstName'
  31. }),
  32. // 带有命名空间的子模块也可以这样实现映射,在方法多的时候可以简化代码
  33. // ...mapGetters('a/c', [
  34. // 'cFirstName'
  35. // ])
  36. },
  37. methods: {
  38. // 映射mutations到方法
  39. ...mapMutations({
  40. modifyBName: 'b/changeBName'
  41. }),
  42. // 也可以这样实现
  43. // ...mapMutations('b', {
  44. // modifyBName: 'changeBName'
  45. // }),
  46. // 带有命名空间的子模块映射到组件的方法
  47. ...mapActions('a', {
  48. modifyCName: 'callChangeCNameAsync'
  49. }),
  50. }
  51. }
  52. </script>
  53. <style>
  54. #app {
  55. margin-top: 60px;
  56. }
  57. </style>

3. 组件之间共享数据

上面说明了Vuex的使用方法,下面看下Vuex在组件共享数据场景的一个简单示例。

有两个子组件bChild和cChild,它们直接从store中获取数据并渲染。在根组件App.vue中修改store中的数据,可以看到子组件会相应数据更新,展示最新的数据。

  1. <template>
  2. <div>
  3. {{counter}}
  4. {{counter10times}}
  5. <b-child></b-child>
  6. <c-child></c-child>
  7. <button @click="modifyBName">修改bname</button>
  8. <button @click="modifyCName">修改cname</button>
  9. </div>
  10. </template>
  11. <script>
  12. import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'
  13. import bChild from './components/bChild';
  14. import cChild from './components/cChild';
  15. export default {
  16. computed: {
  17. ...mapState({
  18. counter: state => state.counter,
  19. // 也可以这样实现
  20. // counter: 'counter',
  21. }),
  22. // 全局的getters
  23. ...mapGetters(['counter10times']),
  24. // 也可以这样实现,指定组件中的数据名称
  25. // ...mapGetters({
  26. // counter10times: 'counter10times'
  27. // }),
  28. },
  29. components: {
  30. 'b-child': bChild,
  31. 'c-child': cChild
  32. },
  33. methods: {
  34. // 映射mutations到方法
  35. ...mapMutations({
  36. modifyBName: 'b/changeBName'
  37. }),
  38. // 也可以这样实现
  39. // ...mapMutations('b', {
  40. // modifyBName: 'changeBName'
  41. // }),
  42. // 带有命名空间的子模块映射到组件的方法
  43. ...mapActions('a', {
  44. modifyCName: 'callChangeCNameAsync'
  45. }),
  46. }
  47. }
  48. </script>
  49. <style>
  50. #app {
  51. margin-top: 60px;
  52. }
  53. </style>

五、Vuex原理

1. 说明

Vuex通过createStore创建了一个数据中心,然后通过发布-订阅模式来让订阅者监听到数据改变。

那么Vuex是怎么应用到Vue中的呢?

先来看一个在Vue中使用Vuex的简单例子:

  1. // main.js
  2. import { createApp } from 'vue';
  3. import App from './App.vue';
  4. import {createStore} from 'vuex';
  5. const store = createStore({
  6. state: {
  7. message: 'hello'
  8. },
  9. mutations: {
  10. change(state) {
  11. state.message = 'world';
  12. }
  13. }
  14. });
  15. createApp(App).use(store).mount('#app');
  1. export default {
  2. name: 'App',
  3. computed: {
  4. info() {
  5. return this.$store.state.message;
  6. }
  7. },
  8. mounted() {
  9. this.$store.commit('change');
  10. }
  11. }

可以看到,在Vue中使用Vuex,主要有3个关键步骤:

  1. 使用Vuex创建store,再将store注入Vue中。Vue组件中就可以通过this.$store来访问到store。
  2. Vue使用computed获取$store中的状态。
  3. Vue通过$store.commit和$store.action来修改状态。

那么我们需要问两个问题:

  1. 注入的原理是什么?为什么调用use()方法之后,就可以在组件通过$store来访问store了?
  2. 响应式原理是什么?为什么使用computed可以监听到store中的状态改变?

这两个是Vuex比较核心的两个原理。

2. 注入原理

store注入 vue的实例组件的方式,是通过vue的 mixin机制,借助vue组件的生命周期钩子beforeCreate 完成的。
  1. Vue.mixin({
  2. beforeCreate() {
  3. if (this.$options && this.$options.store) {
  4. // 找到根组件 main 上面挂一个$store
  5. this.$store = this.$options.store;
  6. }
  7. else {
  8. // 非根组件指向其父组件的$store
  9. this.$store = this.$parent && this.$parent.$store;
  10. }
  11. }
  12. });

3. Vuex响应式原理

Vuex使用vue中的reactive方法将state设置为响应式,原理和Vue组件的data设置为响应式是一样的。

  1. // vuex/src/store-util.js
  2. import {reactive} from 'vue';
  3. store._state = reactive({
  4. data: state
  5. });

4. 总结

Vuex是个状态管理器。

它Vuex通过createStore创建了一个数据中心,然后通过发布-订阅模式来让订阅者监听到数据改变。

Vuex的store注入 vue的实例组件的方式,是通过vue的 mixin机制,借助vue组件的生命周期钩子beforeCreate 完成的。这样Vue组件就能通过this.$store获取到store了。

Vuex使用vue中的reactive方法将state设置为响应式,这样组件就可以通过computed来监听状态的改变了。