zoukankan      html  css  js  c++  java
  • vuex的基础使用整理

    Vuex整理

    Vuex的产生的原因:

    一般的项目中的数据与视图的关系都是“单向数据流”

    即:state在view中展示、action进行state中的状态变化、view中用户进行操作触发action

     

    当多个组件共享同一个状态的时候,单向数据流的简洁模式就会被破坏。可以使用下边的方式解决这些问题

    常规的组件传值的方式:

    当组件的嵌套关系不是很复杂的时候可以使用常规的传值实现状态的共享。

    之前有过记录https://www.cnblogs.com/wyongz/p/11425433.html

    Vue总线机制

    vue中复杂嵌套组件之间通信除了使用vuex,也可以通过bus总线,两者适用场景不同。

    bus适合小项目、数据被更少组件使用的项目,对于中大型项目 数据在很多组件之间使用的情况 bus就不太适用了。bus其实就是一个发布订阅模式,利用vue的自定义事件机制,在触发的地方通过$emit向外发布一个事件,在需要监听的页面,通过$on监听事件。

    vuex适用中大型项目、数据在多组件之间公用的情况。

    这种方法的原理是通过一个空的Vue实例作为中央事件总线,通过它来触发事件和监听事件,可以实现几乎任何组件间的通信

      在utils文件夹下面定义一个bus.js,这样一个中央事件总线就定好了,里面内容就这两行:

    import Vue from 'vue'
    
    export const Event = new Vue() // 定义一个事件总线

    传递方:

    import { Event } from '../../utils/bus'  // 引入这个事件总线
    
    export default {
    
      data() {
    
        userId: ''
    
      }
    
      watch: {
    
        userId() {  // 监听userId改变时触发事件,当然你可以在所有其他事件去触发$emit
    
          Event.$emit('changeUser', userId) // $emit接收参数,一个是触发的事件名,一个是携带的参数
    
        }
    
    }

    接收方:

    import { Event } from '../../utils/bus' // 依然是把bus总线引进来
    
    export default {
    
    created() {
    
        Event.$on('changeSonRoute',showIndex => {  // $on接收发送方传给我们的数据
    
          console.log(showIndex)
    
        })
    
      }

    综上所述,在大项目中组件嵌套比较复杂的情况下的状态共享使用vuex比较合适

    Vuex的使用

     

     Vuex 的状态存储是响应式的。当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会相应地得到高效更新。

    首先引入vuex并使用

    import Vuex from 'vuex'
    
    Vue.use(Vuex)

    创建实例

    const store = new Vuex.Store({
    
      state: {
    
        count: 0
    
      },
    
      mutations: {
    
        increment (state) {
    
          state.count++
    
        }
    
      }
    
    })

    注入vue中,这样就可以通过实例调用了this.$store

    new Vue({
    
      el: '#app',
    
      store,
    
      render: h => h(App)
    
    })

    一般规范是State中存放状态,multation中修改状态,action中存放业务逻辑

    State

      改变 store 中的状态的唯一途径就是显式地提交 (commit) mutation。这样使得我们可以方便地跟踪每一个状态的变化

    子组件中访问state

      上边通过在根实例中注册 store 选项, store 实例会注入到根组件下的所有子组件中,所以子组件能通过 this.$store 访问到

      由于 Vuex 的状态存储是响应式的,使用状态最简单的方法就是在计算属性中返回某个状态,当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
        }
      })
    }

    三种写法:

    1、 箭头函数

    2、 直接写状态中的名称(字符串形式)

    3、 如果需要vuex中的状态与组件中的状态进行计算属性的要使用常规函数

    Vuex中的状态与组件中的计算属性混合使用

    因为mapState是对象形式,可以使用解构的方式获取对象中的属性

    computed: {
      localComputed () { /* ... */ },
      // 使用对象展开运算符将此对象混入到外部对象中
      ...mapState({
        // ...
      })
    }

    Getter

    Vuex中的getter相当于组件中的计算属性,从 store 中的 state 中派生出一些状态

    Getter用法:

    Getter 接受 state 作为其第一个参数

    getters: {
        doneTodos: state => {
          return state.todos.filter(todo => todo.done)
        }
      }

    如果返回的是属性的话使用方式和state相同,但是如果需要外部传值的就要返回方法并在使用的时候传入需要的参数

    属性

    computed: {
      doneTodosCount () {
        return this.$store.getters.doneTodosCount
      }
    }

    方法

    getters: {
      // ...
      getTodoById: (state) => (id) => {
        return state.todos.find(todo => todo.id === id)
      }
    }
    store.getters.getTodoById(2)

    这种方式有点类似于react中的高阶函数

    mapGetters 辅助函数

    import { mapGetters } from 'vuex'
    export default {
      // ...
      computed: {
      // 使用对象展开运算符将 getter 混入 computed 对象中
        ...mapGetters([
          'doneTodosCount',
          'anotherGetter',
    doneCount: 'doneTodosCount'//不使用getter中的名字另取名字的写法
          // ...
        ])
      }
    }

    Mutation和action的区别

    下边涉及到mutation和action,所以在这插入一下两者之间的区别

    1、流程顺序

    “相应视图—>修改State”拆分成两部分,视图触发Action,Action再触发Mutation。

    2、角色定位

    基于流程顺序,二者扮演不同的角色。

    Mutation:专注于修改State,理论上是修改State的唯一途径。

    Action:业务代码、异步请求。

    3、限制

    角色不同,二者有不同的限制。

    Mutation:必须同步执行。

    Action:可以异步,但不能直接操作State

    Mutation

    更改 Vuex 的 store 中的状态的唯一方法是提交 mutation

    定义:
    mutations: {
        increment (state) {
          // 变更状态
          state.count++
        }
      }
    调用:
    store.commit('increment')

    向 store.commit 传入额外的参数

    mutations: {

      increment (state, n) {

        state.count += n

      }

    }

    store.commit('increment', 10)

    参数应该是一个对象,这样可以包含多个字段并且记录的 mutation 会更易读

    对象形式的调用:

    store.commit({
      type: 'increment',
      amount: 10
    })

    Mutation 需遵守 Vue 的响应规则

    最好提前在你的 store 中初始化好所有所需属性。

    当需要在对象上添加新属性时,你应该

    使用 Vue.set(obj, 'newProp', 123), 或者以新对象替换老对象

    在组件中需要多个提交 Mutation

    import { mapMutations } from 'vuex'
    export default {
      // ...
      methods: {
        ...mapMutations([
          'increment', // 将 `this.increment()` 映射为 `this.$store.commit('increment')`
          // `mapMutations` 也支持载荷:
          'incrementBy' // 将 `this.incrementBy(amount)` 映射为 `this.$store.commit('incrementBy', amount)`
        ]),
        ...mapMutations({
          add: 'increment' // 将 `this.add()` 映射为 `this.$store.commit('increment')`
        })
      }
    }

    Action

    Action中处理业务逻辑,来使mutation中保持只进行state的状态修改操作

    而且mutation中不能执行异步的操作,可以在action中执行异步操作

    你在组件中使用 this.$store.dispatch('xxx') 分发 action,或者使用 mapActions 辅助函数将组件的 methods 映射为 store.dispatch 调用

    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')`
        })
      }
    }

    组合 Action

    Action 通常是异步的,那么如何知道 action 什么时候结束呢?更重要的是,我们如何才能组合多个 action,以处理更加复杂的异步流程?

    store.dispatch 可以处理被触发的 action 的处理函数返回的 Promise,并且 store.dispatch 仍旧返回 Promise:

    actions: {
      actionA ({ commit }) {
        return new Promise((resolve, reject) => {
          setTimeout(() => {
            commit('someMutation')
            resolve()
          }, 1000)
        })
      }
    }

    现在你可以:

    store.dispatch('actionA').then(() => {

      // ...

    })

    在另外一个 action 中也可以:

    actions: {

      // ...

      actionB ({ dispatch, commit }) {

        return dispatch('actionA').then(() => {

          commit('someOtherMutation')

        })

      }

    }

    最后,如果我们利用 async / await,我们可以如下组合 action:最典型的使用就是用户权限信息的判断部分,等后台返回数据后再进行判断来使用vuex来存储登录信息

    // 假设 getData() 和 getOtherData() 返回的是 Promise
    actions: {
      async actionA ({ commit }) {
        commit('gotData', await getData())
      },
      async actionB ({ dispatch, commit }) {
        await dispatch('actionA') // 等待 actionA 完成
        commit('gotOtherData', await getOtherData())
      }
    }
  • 相关阅读:
    闲谈系列之一——数据库主键GUID
    一个简单通用权限管理系统,求各位帮忙看看
    php 计算指定年份的周总数与及第几周的开始日期和结束日期(从周一开始)
    创建虚拟机流程详细过程链接
    阿里云CDN加速设置
    sublime Text3 快捷键
    Linux命令(centos7)
    分布式数据库
    mysql 分区和分表
    Linux crontab 命令格式与详细例子
  • 原文地址:https://www.cnblogs.com/wyongz/p/12923850.html
Copyright © 2011-2022 走看看