vuex的使用

2022-06-301670

[TOC]

Vuex

了解Vuex

Vuex是一个专门为Vue.js应用程序开发的全局状态管理模式, 它采用集中式存储管理所有组件的公共状态, 并以相应的规则保证状态以一种可预测的方式发生变化

为什么要使用Vuex

vue组件是Vue最强大的功能之一,所以要掌握Vue,组件间的通信是必须掌握的技能,但如果你使用 vue.js 开发过大型项目, 那么我想你可能会对 vue 组件之间的通信感到崩溃 。

vuex的出现就是为了解决多组件间的数据通讯(下图为:Vuex的核心工作原理)

vuex

使用

步骤

  1. 引入
  2. 安装插件
  3. 实例化store
  4. 将store实例注入到Vue根实例
  5. 在组件中使用
    import Vue from 'vue'
    // 1. 引入
    import Vuex from 'vuex'、

    // 2. 安装插件
    Vue.use(Vuex)

    // 3.创建store
    const store = new Vuex.Store({
        // 配置核心参数
        state:{
            products:[
                {name: '鼠标', price: 20},
                {name: '键盘', price: 40},
                {name: '耳机', price: 60},
                {name: '显示屏', price: 80}
            ]
        }
    }

    // 4. store注入到vue实例
    new Vue({
        // ... 
        store
    });

    // 5. 在myapp组件中使用
    this.$store.state.products

Vuex的核心配置

  • state

    state就是Vuex中的公共的状态, 用于保存所有组件的公共数据,类似于组件中的data

    • 获取:this.$store.state.xxx
        state:{
            products:[
                {id:1, name: '鼠标', price: 20},
                {id:2, name: '键盘', price: 40},
                {id:3, name: '耳机', price: 60},
                {id:4, name: '显示屏', price: 80}
            ],
        },
  • getters

    类似于组件中的computed,getter 的返回值会根据它的依赖被缓存起来,且只有当它的依赖值发生了改变才会被重新计算。

    • 获取:this.$store.getters.xxx
        getters:{
            saleProducts(state,getters,rootState,rootGetters){
                // 无模块化时:state===rootState
                // 无命名空间时:getter===rootGetters
                return state.products.map(item=>{
                    return {
                        price:item.price/2,
                        name:item.name
                    }
                });
            }
        },
  • mutations

    修改state的唯一方式,类似于组件中的methods

    • mutation参数

      • state
      • payload 触发函数时传入的参数(只允许传递一下参数)
    • 调用方式:this.$store.commit(type,payload)

        mutations:{
            // 定义
            change(state,payload){
                state.products = state.products.map(item=>{
                    return {...item,...payload};
                });
            }
        }
    
        // 调用
        this.$store.commit('change',{price:88})
  • actions

    类似于 mutations,负责异步操作(actions中可以包含异步操作, mutations中绝对不允许出现异步),actions用来操作mutations,而mutations用来操作state

    • action参数
      • context: 与store实例具有相同属性和方法的对象
      • payload 触发函数时传入的参数
    • 调用方式:this.$store.dispatch(type,payload)
        actions:{ 
            // 定义
            minusPriceAsync( context, payload ) {
                setTimeout( () => {
                    context.commit( 'change', payload ); //context提交
                }, 2000)
            }
        }
    
        // 调用
        this.$store.dispatch('minusPriceAsync',{price:100})

store模块化

由于使用单一状态state(即所有状态会集中到state这个对象),当应用变得非常复杂时,store 对象就有可能变得相当臃肿。为了解决以上问题,Vuex 允许我们将 store 分割成模块(module)。每个模块拥有自己的 state、mutation、action、getter等,这样好管理又容易维护,模块化后,默认只影响state的获取,getters,mutations,actions还是保存在全局状态

  • 模块内部的 mutation 和 getter,接收的第一个参数state是模块的局部状态对象
  • action中
    • context.state 局部状态
    • context.rootState 全局状态
    import Vue from 'vue';
    import Vux from 'vuex';

    // 引入其他store模块
    import home from './home.js';
    import top250 from './top250.js';

    export default new Vuex.Store({
        modules:{
            home,
            top250
        }
    });

    // home.js
    const home = {
        state:{
            numbers:[10,6,3,9,12,18,5]
        },
        getters:{
            small(state,getters,rootState,rootGetters){
                return state.numbers.filter(item=>item<10)
            }
        },
        mutations:{

        }
    }
    export default home
  • namespace命名空间

    namespaced:true,设置命名空间后让getters,mutations,actions具有自己的命名空间

    模块化之后默认只有state的获取改变$store.state.模块名.XXX,
    命名空间是在模块化之后才能使用的,因为他是设置在模块里面的,
    {
        namespaced: true
        state: {},
        mutations: {}
    }
    ,获取和state模块化后的获取略有不同
    $store.getters['模块名/XXX']

映射Vuex

使用mapState,mapGetters,mapMutations,mapActions把state,getters,mutations,actions映射到组件中使用,主要用于简化操作

    // 模块化开发中直接引入mapState使用
    import {mapState,mapGetters,mapMutations,mapActions} from 'vuex';

    computed:{
        // 数组形式
        ...mapState([
            'products' //映射computed.products为this.$store.state.products
        ]),
        // 对象形式
        ...mapState({
            // products:'products',
            // products:state=>state.products,
            products(state){
                return state.products;
            }
        }),
        ...mapGetters([]),
        // ...mapGetters(namespace,{}),// 只有设置了命名空间才可以使用第一个参数
    },
    methods:{
        // 数组形式
        ...mapMutations([
            'change',//映射this.change为this.$store.commit('change')
            ]),
        // 对象形式
        ...mapMutations({
            change2:'change',//映射this.change2为this.$store.commit('change')

            // 函数形式:
            change3:(commit,payload)=>{
                commit('change',payload);
            }
        }),
        ...mapActions({
            modify:'minusPriceAsync', //映射this.modify()为this.$store.dispatch('minusPriceAsync'),

            // 函数形式:自动传入dispatch,payload为调用,minus是传入的参数
            minus(dispatch,payload){
                dispatch('minusPriceAsync',payload)
            }
        })
    }
分享
点赞0
打赏
上一篇:代理工具Fiddler -调试与替换接口状态
下一篇:vben框架-快速上手开发