【Vue】使用 Vuex 作为状态管理
Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式和库。它使用单一状态树,这意味着这个对象包含了全部的应用层级状态,并且以一种相对集中的方式存在。这也意味着,通常单个项目中只有一个 Vuex store。Vuex 的核心概念和功能包括:
-
状态(State):Vuex 使用单一状态树,即一个对象包含了整个应用的状态。状态存储是响应式的,当 Vue 组件从 store 中读取状态时,若状态发生变化,相关组件也会相应更新。
-
视图(View):Vue 组件输出状态的可视化表示。
-
行为(Actions):响应在 view 上的用户输入导致的状态变化。
除此之外,Vuex 还有以下几个核心概念:
-
Getters:允许组件从 store 中获取状态,同时可以对状态进行一些预处理。
-
Mutations:更改 Vuex 的 store 中的状态的唯一方法是提交 mutation。Vuex 中的 mutations 非常类似于事件:每个 mutation 都有一个字符串的事件类型 (type) 和一个回调函数 (handler)。
-
Actions:类似于 mutations,不同在于它们提交的是 mutations,而不是直接变更状态。Actions 可以包含任意异步操作。
-
Modules:由于使用单一状态树,应用的所有状态会集中到一个比较大的对象。当应用变得非常复杂时,store 对象就有可能变得相当臃肿。为了解决这个问题,Vuex 允许我们将 store 分割成模块(module)。
Vuex 是响应式的。当 Vue 组件从 store 中读取状态的时候,若 store 中的状态发生变化,那么相应的组件也会自动更新。这也意味着 Vuex 中的状态不能直接改变,只能通过显式提交 mutations 来更改。
使用案例
假设我们正在开发一个购物车应用,用户可以将商品添加到购物车中,查看购物车中的商品,以及清空购物车。在这个例子中,我们将创建一个 Vuex store 来管理购物车的状态。
首先,你需要在你的 Vue 项目中安装和引入 Vuex:
pnpm install vuex --save
然后,在你的 Vue 项目中创建一个 Vuex store:
// store.js import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) export default new Vuex.Store({ state: { cart: [] }, getters: { cartItemCount: state => { return state.cart.length }, cartTotalPrice: state => { return state.cart.reduce((total, item) => { return total + item.price * item.quantity }, 0) } }, mutations: { addToCart(state, item) { const found = state.cart.find(product => product.id === item.id) if (found) { found.quantity++ } else { state.cart.push({ ...item, quantity: 1 }) } }, removeFromCart(state, item) { const index = state.cart.indexOf(item) if (index > -1) { state.cart.splice(index, 1) } }, clearCart(state) { state.cart = [] } }, actions: { addToCart({ commit }, item) { commit('addToCart', item) }, removeFromCart({ commit }, item) { commit('removeFromCart', item) }, clearCart({ commit }) { commit('clearCart') } } })
在上面的代码中,我们定义了:
- State:购物车数组 cart。
- Getters:cartItemCount 和 cartTotalPrice 用于获取购物车中的商品数量和总价。
- Mutations:addToCart、removeFromCart 和 clearCart 用于修改购物车的状态。
- Actions:包装了 mutations 的方法,这里是 addToCart、removeFromCart 和 clearCart。
最后,在你的 Vue 组件中使用这个 store:
// 在一个组件中 <script> import { mapGetters, mapActions } from 'vuex' export default { computed: { ...mapGetters(['cartItemCount', 'cartTotalPrice']) }, methods: { ...mapActions(['addToCart', 'removeFromCart', 'clearCart']), addItemToCart(item) { this.addToCart(item) }, removeItemFromCart(item) { this.removeFromCart(item) }, emptyCart() { this.clearCart() } } } </script>
完整代码
<template> <div class="shopping-cart"> <h2>购物车</h2> <div v-if="cartItemCount === 0"> 购物车为空。 </div> <div v-else> <ul> <li v-for="(item, index) in cart" :key="index"> {{ item.name }} - {{ item.quantity }} x {{ item.price }}元 <button @click="removeItemFromCart(item)">移除</button> </li> </ul> <p>总价: {{ cartTotalPrice }}元</p> <button @click="emptyCart">清空购物车</button> </div> </div> </template> <script> import { mapGetters, mapActions } from 'vuex' export default { name: 'ShoppingCart', computed: { ...mapGetters(['cartItemCount', 'cartTotalPrice']), cart() { return this.$store.state.cart } }, methods: { ...mapActions(['removeFromCart', 'clearCart']), removeItemFromCart(item) { this.removeFromCart(item) }, emptyCart() { this.clearCart() } } } </script> <style scoped> .shopping-cart { border: 1px solid #ddd; padding: 20px; border-radius: 5px; margin: 20px auto; width: 300px; } .shopping-cart h2 { color: #333; } .shopping-cart ul { list-style-type: none; padding: 0; } .shopping-cart ul li { margin-bottom: 10px; line-height: 1.6; } .shopping-cart button { background-color: #42b983; color: white; border: none; padding: 5px 10px; margin-left: 10px; border-radius: 3px; cursor: pointer; } .shopping-cart button:hover { background-color: #333; } </style>