VueX总结(一)

1.介绍

首先Vuex是Vue的核心插件(还没了解vue的可以到 vue官网 学习下),所以使用VueX之前必须有Vue,是用来集中存储组件状态(也就是数据).我们需要搞清楚一点项目中是不是真的需要用vuex,不要为了使用vuex而使用vuex

2.基础使用及注册

vuex.js下载地址

<script src="/path/to/vue.js"></script>
<script src="/path/to/vuex.js"></script>

npm 

npm install vuex --save
import Vuex from 'vuex'
import Vue from 'vue'
Vue.use(Vuex);
//gitHub dome查看

const moduleA = {
  state: { count: 100 },
  mutations: {
    increment (state) {
      // 这里的 `state` 对象是模块的局部状态
      state.count++
    }
  },

  getters: {
    doubleCount (state, getters, rootState) {
      return state.count * 2 +rootState.count;
    }
  }
};

export default new Vuex.Store({
  state(){  //存放数据
    return {  //防止污染
      count: 0,
      str: 'cas'
    }
  },
  getters:{  //state的计算属性
    doneTodos: state => {
      return state.str+'C';
    }
  },
  mutations:{  //同步
    /**
     * @param state state中注册值
     * @paramList 自行添加至
     */
    increment(state){
      state.count++
    },
    increment1(state){
      state.count++
    },

  },
  actions:{  //异步 (调用接口)
    increment(context){
      context.commit('increment');
    },
    increment1({commit}){
      console.log(commit)
      commit('increment1');
    }
  },
  modules:{  //模块的局部状态 可以按模块组件划分
    a:moduleA,
    account: {
      namespaced: true,
      //允许子模型对应方法注册再自己的路径下,防止vuex中有同名的方法 调用一起触发
      // this.$store.dispatch('account/login')
      // this.$store.dispatch('account/login1')
      // this.$store.dispatch('account/posts/login2')

      state: {
        account:'account'
      }, // 模块内的状态已经是嵌套的了,使用 `namespaced` 属性不会对其产生影响
      actions: {
        login ({ state, commit, rootState }) {
          commit('login')
        }
      },
      mutations: {
        login (state) {
          console.log('account'+state.account)
        },
        incrementcccc () {
          console.log('accountincrement')
        }
      },
      // 嵌套模块
      modules: {
        // 继承父模块的命名空间
        myPage: {
          state: {
            dome1:'21',
          },
          actions: {
            login1 ({ state, commit, rootState }) {
              commit('login1')
            }
          },
          mutations: {
            login1 (state) {
              console.log('myPage'+state.dome1)
            },
          },
        },
        // 进一步嵌套命名空间
        posts: {
          namespaced: true,
          state: {
            dome1:'21',
            dome2:'22',
          },
          actions: {
            login2 ({ state, commit, rootState }) {
              commit('login2');
              commit('account/incrementcccc',null,{root:true})   //root用来掉用别的模块的方法
            }
          },
          mutations: {
            login2 (state) {
              console.log('posts'+state.dome1)
            },
          },
        },
        foo: {
          namespaced: true,
          actions: {
            someAction: {
              root: true,   //可以注册全局的无视路径调用
              handler (namespacedContext, payload) {
                console.log('外部调用')
              } // -> 'someAction'
            }
          }
        }
      }
    }
  }
})

上面是我随便写的store文件,我先使用webpack搭建了个项目,然后使用 import导入 new Vuex.store创建vuex,然后到new Vue 的store里注册.这里我把store单独提到了一个文件里,通过export defaut讲整个Vuex.store抛出去,然后import到main.js注册,这样可以防止mian代码过多,各个功能模块之间解耦合.

3.state

用来存储数据的就像vue中的data,这里有个小点就是不要state:{}里直接写对象最好return{},防止数据被随意或不小心修改而污染

 state(){  //存放数据
    return {  //防止污染
      count: 0,
      str: 'cas'
    }
  },

mapState 辅助函数

state中有个辅助函数,是将state中的对象映射的vue的计算属性中,有人会问mapState怎么来,这个vuex已经暴露出来了 我们只需要 import {mapState} from 'vuex'

 computed:{
    computeddeom1:function () {
      return this.deom1+3;
    },
    ...mapState({  //计算数据可以直接调用 vuex state 中数据
      count:state => state.count,
      conutAlias :'count',
    }),

    //同个模块下多个相同的
    ...mapState({
      a:state => state.account.posts.dome1,
      d:state => state.account.posts.dome2,
    }),
    ...mapState('account/posts',{
      b:state => state.dome1,
      c:state => state.dome2,
    })
},

4.getter

getter其实也很简单就想到与vue的computed计算属性,this.$store.getters.doneTodos调用

getters:{  //state的计算属性
    doneTodos: state => {
      return state.str+'C';
    }
  },

mapGetters 辅助函数和mapState一样就是一个映射的是Getter,这里我就不讲了

5.mutation

mutation是用来改变store中数据值得唯一方式,而且mutation中只能执行同步操作不能执行异步

 mutations:{  //同步
    /**
     * @param state state中注册值
     * @paramList 自行添加至
     */
    increment(state){
      state.count++
    },
    increment1(state,data,...){
      state.count++
    },

  },



this.$store.commit('increment')
如果有多个参数就
this.$store.commit('increment',a,b,....)

6.action

action是用来提交mutation,不是直接改变store值得,就类似于代理设计模式,就是在mutation方法上包装增强一下,而且action中是可以做异步操作的.

  actions:{  //异步 (调用接口)
    increment(context){
      context.commit('increment');
    },
    increment1({commit}){
      console.log(commit)
      commit('increment1');
    }
  },


this.$store.dispatch('increment')  调用

mapActions 辅助函数

mapAction和mapGetter mapState原理一样就是先将action中方法映射到 vue 的 methods中

这段是官网代码
import { mapActions } from 'vuex'

export default {
  // ...
  methods: {
    ...mapActions([
      'increment', // 将 `this.increment()` 映射为 `this.$store.dispatch('increment')`

      // `mapActions` 也支持载荷:
      'incrementBy' // 将 `this.incrementBy(amount)` 映射为 `this.$store.dispatch('incrementBy', amount)`
    ]),
    ...mapActions({
      add: 'increment' // 将 `this.add()` 映射为 `this.$store.dispatch('increment')`
    })
  }
}

7.module

module就是一个 state,getter,mutation,action,module的总和,对于轻型项目没必要使用module,把所有的状态都写到state中就行了

大型或逻辑复杂的项目模块组件多,如果都写在store下的state中的话会变得相当臃肿,耦合性强,不便于后期迭代和维护

  modules:{  //模块的局部状态 可以按模块组件划分
    a:moduleA,
    account: {
      namespaced: true,
      //允许子模型对应方法注册再自己的路径下,防止vuex中有同名的方法 调用一起触发
      // this.$store.dispatch('account/login')
      // this.$store.dispatch('account/login1')
      // this.$store.dispatch('account/posts/login2')

      state: {
        account:'account'
      }, // 模块内的状态已经是嵌套的了,使用 `namespaced` 属性不会对其产生影响
      actions: {
        login ({ state, commit, rootState }) {
          commit('login')
        }
      },
      mutations: {
        login (state) {
          console.log('account'+state.account)
        },
        incrementcccc () {
          console.log('accountincrement')
        }
      },
      // 嵌套模块
      modules: {
        // 继承父模块的命名空间
        myPage: {
          state: {
            dome1:'21',
          },
          actions: {
            login1 ({ state, commit, rootState }) {
              commit('login1')
            }
          },
          mutations: {
            login1 (state) {
              console.log('myPage'+state.dome1)
            },
          },
        },
        // 进一步嵌套命名空间
        posts: {
          namespaced: true,
          state: {
            dome1:'21',
            dome2:'22',
          },
          actions: {
            login2 ({ state, commit, rootState }) {
              commit('login2');
              commit('account/incrementcccc',null,{root:true})   //root用来掉用别的模块的方法
            }
          },
          mutations: {
            login2 (state) {
              console.log('posts'+state.dome1)
            },
          },
        },
        foo: {
          namespaced: true,
          actions: {
            someAction: {
              root: true,   //可以注册全局的无视路径调用
              handler (namespacedContext, payload) {
                console.log('外部调用')
              } // -> 'someAction'
            }
          }
        }
      }
    }
  }

module可以直接account:{}  也 可以import 对象 a:module 

这里我建议第二张这种每个模块的module都在一个单独的js里这种代码会简洁易读

命名空间

namespaced: true,  
允许子模型对应方法注册再自己的路径下,防止vuex中有同名的方法 调用一起触发

在带命名空间的模块内容访问全局内容

简单的说就是调用不懂module中的方法和值

commit('account/incrementcccc',null,{root:true})
    posts: {
          namespaced: true,
          state: {
            dome1:'21',
            dome2:'22',
          },
          actions: {
            login2 ({ state, commit, rootState }) {
              commit('login2');
              commit('account/incrementcccc',null,{root:true})   //root用来掉用别的模块的方法
            }
          },
          mutations: {
            login2 (state) {
              console.log('posts'+state.dome1)
            },
          },
        },
        foo: {
          namespaced: true,
          actions: {
            someAction: {
              root: true,   //可以注册全局的无视路径调用
              handler (namespacedContext, payload) {
                console.log('外部调用')
              } // -> 'someAction'
            }
          }
        }

带命名空间的绑定函数

就是使用辅助函数按对应模块映射

  computed:{
    computeddeom1:function () {
      return this.deom1+3;
    },
    ...mapState({  //计算数据可以直接调用 vuex state 中数据
      count:state => state.count,
      conutAlias :'count',
    }),

    //同个模块下多个相同的
    ...mapState({
      a:state => state.account.posts.dome1,
      d:state => state.account.posts.dome2,
    }),
    ...mapState('account/posts',{
      b:state => state.dome1,
      c:state => state.dome2,
    })
},

也可以通过createNamespacedHelpers创建对应模块的命名函数

/**
 * createNamespacedHelpers 带命名空间的绑定函数
 */
import { createNamespacedHelpers } from 'vuex'

const { mapState, mapActions } = createNamespacedHelpers('account/posts')

export default {
    // 在 `some/nested/module` 中查找
    ...mapState({
      a: state => state.dome1,
      b: state => state.dome2
    })
}

模块动态注册

    // 注册模块 `myModule` (动态加载管理) vuex-router-sync
    //store.unregisterModule(moduleName)  注销
    //store.registerModule('a', module, { preserveState: true })  preserveState 是否让浏览器保存
    store.registerModule('myModule', {
      state: {
        data1:'registerModule1',
      },
    })
    store.registerModule(['myModule', 'nested'], {
      state: {
        data1:'registerModule2',
      },
    })
    console.log(this.$store.state.myModule.data1+'-registerModule')
    console.log(this.$store.state.myModule.nested.data1+'-registerModule')

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值