当前位置: 首页 > article >正文

vue-20(Vuex 状态管理的最佳实践)

Vuex 状态管理的最佳实践

Vuex 是管理大型 Vue.js 应用状态的一个强大工具,但其有效性取决于其组织和维护的质量。管理不善的 Vuex 存储可能会变得难以控制、难以调试,并成为性能瓶颈。本课程深入探讨构建 Vuex 存储的最佳实践,重点关注可维护性、可扩展性和性能。我们将探讨组织状态、操作、变异和获取器的技术,以及处理异步操作和优化性能的策略。

为可扩展性构建 Vuex 存储

随着应用程序的增长,单个 Vuex 存储很快就会变得令人不知所措。模块是组织存储为可管理部分的关键。

使用命名空间进行模块化

Vuex 模块允许你将你的 store 分割成更小、自包含的单元。命名空间对于防止命名冲突和改进代码组织至关重要。

  • 基础模块结构:

    // store/modules/user.js
    export default {namespaced: true,state: () => ({profile: null,isLoggedIn: false}),mutations: {SET_PROFILE (state, profile) {state.profile = profilestate.isLoggedIn = true},CLEAR_PROFILE (state) {state.profile = nullstate.isLoggedIn = false}},actions: {async fetchProfile ({ commit }, userId) {// Simulate API callawait new Promise(resolve => setTimeout(resolve, 500));const profile = { id: userId, name: 'John Doe', email: 'john.doe@example.com' };commit('SET_PROFILE', profile);},logout ({ commit }) {commit('CLEAR_PROFILE');}},getters: {fullName: (state) => state.profile ? `${state.profile.name}` : null,isLoggedIn: (state) => state.isLoggedIn}
    }
    

    在这个例子中,user 模块管理与用户相关的状态。namespaced: true 确保所有操作、变更和获取器都通过 user/ 前缀访问。

  • 在store中注册模块:

    // store/index.js
    import Vue from 'vue'
    import Vuex from 'vuex'
    import user from './modules/user'
    import products from './modules/products'Vue.use(Vuex)export default new Vuex.Store({modules: {user,products}
    })
    

    用户 模块已注册在主 Vuex 存储中。现在,您可以使用 store.dispatch('user/fetchProfile', userId) 访问 获取个人资料 操作。

  • 在组件中访问命名空间模块:

    <template><div><p v-if="isLoggedIn">Welcome, {{ fullName }}!</p><button @click="fetchUserProfile(123)">Fetch Profile</button><button @click="logoutUser">Logout</button></div>
    </template><script>
    import { mapGetters, mapActions } from 'vuex';export default {computed: {...mapGetters('user', ['fullName', 'isLoggedIn'])},methods: {...mapActions('user', ['fetchProfile', 'logout']),fetchUserProfile(userId) {this.fetchProfile(userId);},logoutUser() {this.logout();}}
    }
    </script>
    

    Vuex 中的 mapGettersmapActions 辅助函数用于访问 user 模块中的 fullNameisLoggedInfetchProfilelogout。这些辅助函数的第一个参数是模块的命名空间。

嵌套模块

对于非常大的应用程序,您可能需要将模块嵌套在彼此之中以创建层次结构。

  • 嵌套模块示例:

    // store/modules/settings/index.js
    export default {namespaced: true,modules: {profile: {namespaced: true,state: () => ({theme: 'light',language: 'en'}),mutations: {SET_THEME (state, theme) {state.theme = theme},SET_LANGUAGE (state, language) {state.language = language}},actions: {updateTheme ({ commit }, theme) {commit('SET_THEME', theme)},updateLanguage ({ commit }, language) {commit('SET_LANGUAGE', language)}},getters: {currentTheme: (state) => state.theme,currentLanguage: (state) => state.language}},notifications: {namespaced: true,// ... notification settings module}}
    }// store/index.js
    import settings from './modules/settings'export default new Vuex.Store({modules: {settings}
    })
    

    在这个示例中,settings 模块包含嵌套的 profilenotifications 模块。要访问 updateTheme 操作,您会使用 store.dispatch('settings/profile/updateTheme', 'dark')

动态模块注册

Vuex 允许你动态注册模块,这对于代码拆分或按需加载模块很有用。

  • 动态模块注册示例:

    // In a component
    export default {mounted () {if (!this.$store.state.myModule) {this.$store.registerModule('myModule', {namespaced: true,state: () => ({ count: 0 }),mutations: {increment (state) {state.count++}},actions: {incrementAsync ({ commit }) {setTimeout(() => {commit('increment')}, 1000)}},getters: {doubleCount: (state) => state.count * 2}})}},beforeDestroy () {if (this.$store.state.myModule) {this.$store.unregisterModule('myModule')}},methods: {incrementModuleCount() {this.$store.dispatch('myModule/incrementAsync');}}
    }
    

    这个示例在组件挂载时注册一个名为 myModule 的模块,并在组件销毁时注销它。这对于仅在应用程序某些部分需要的特定功能模块很有用。

操作和变异:异步操作与数据处理

Vuex 中的操作用于处理异步操作,而变异用于同步更新状态。

从actions提交mutations

actions应该提交mutations来更新状态。这确保了所有状态变化都是可追踪和可预测的。

  • 提交变更示例:

    // store/modules/products.js
    export default {namespaced: true,state: () => ({products: []}),mutations: {SET_PRODUCTS (state, products) {state.products = products}},actions: {async fetchProducts ({ commit }) {// Simulate API callawait new Promise(resolve => setTimeout(resolve, 500));const products = [{ id: 1, name: 'Product A' }, { id: 2, name: 'Product B' }];commit('SET_PRODUCTS', products)}},getters: {allProducts: (state) => state.products}
    }
    

    fetchProducts 动作从 API(此处为模拟)获取产品,然后提交 SET_PRODUCTS 变换来更新状态。

使用 Promise 和 Async/Await

操作可以使用 Promise 和 async/await 更清晰地处理异步操作。

  • 使用 Async/Await 的示例:

    // store/modules/products.js
    actions: {async fetchProducts ({ commit }) {try {const response = await fetch('/api/products')const products = await response.json()commit('SET_PRODUCTS', products)} catch (error) {console.error('Error fetching products:', error)// Handle error appropriately, maybe commit another mutationcommit('SET_PRODUCTS_ERROR', error);}}
    }
    

    此示例使用 async/await 从 API 获取产品并处理可能发生的任何错误。

处理动作中的错误

处理操作中的错误并提交变更以相应更新状态是很重要的。

  • 错误处理的示例:

    // store/modules/products.js
    state: () => ({products: [],error: null
    }),
    mutations: {SET_PRODUCTS (state, products) {state.products = productsstate.error = null},SET_PRODUCTS_ERROR (state, error) {state.error = error}
    },
    actions: {async fetchProducts ({ commit }) {try {const response = await fetch('/api/products')const products = await response.json()commit('SET_PRODUCTS', products)} catch (error) {console.error('Error fetching products:', error)commit('SET_PRODUCTS_ERROR', error)}}
    },
    getters: {hasError: (state) => state.error !== null,errorMessage: (state) => state.error
    }
    

    这个示例添加了一个 error 状态属性和一个 SET_PRODUCTS_ERROR 变异来处理在 fetchProducts 操作期间发生的错误。

获取器:派生状态和封装逻辑

获取器用于从存储中派生状态。它们可以用来封装复杂逻辑,并通过缓存结果来提高性能。

使用获取器处理计算属性

获取器应该用于任何依赖于存储状态的计算属性。

  • 使用 Getters 的示例:

    // store/modules/cart.js
    export default {namespaced: true,state: () => ({items: [{ productId: 1, quantity: 2 },{ productId: 2, quantity: 1 }],products: [{ id: 1, name: 'Product A', price: 10 },{ id: 2, name: 'Product B', price: 20 }]}),getters: {cartTotal: (state) => {return state.items.reduce((total, item) => {const product = state.products.find(p => p.id === item.productId)return total + product.price * item.quantity}, 0)},formattedCartTotal: (state, getters) => {return `$${getters.cartTotal.toFixed(2)}`}}
    }
    

    cartTotal 获取器计算购物车中商品的总价值。formattedCartTotal 获取器将总额格式化为货币字符串。

作为函数的 Getter

Getter 也可以返回一个函数,允许你向 Getter 传递参数。

  • 作为函数的 Getter 示例:

    // store/modules/products.js
    export default {namespaced: true,state: () => ({products: [{ id: 1, name: 'Product A', price: 10 },{ id: 2, name: 'Product B', price: 20 }]}),getters: {productById: (state) => (id) => {return state.products.find(product => product.id === id)}}
    }// In a component
    computed: {product() {return this.$store.getters['products/productById'](1);}
    }
    

    productById 获取器返回一个函数,该函数接受一个 id 作为参数,并返回具有该 ID 的产品。

插件:扩展 Vuex 功能

Vuex 插件允许你通过拦截变更和操作来扩展 Vuex 的功能。

使用插件进行日志记录

插件可用于记录变更和操作,这对于调试很有帮助。

  • 日志插件示例:

    const myPlugin = store => {store.subscribe((mutation, state) => {console.log(mutation.type)console.log(mutation.payload)})
    }export default new Vuex.Store({plugins: [myPlugin]
    })
    

    这个插件将每个变更的类型和有效载荷记录到控制台。

使用插件持久化状态

插件可用于将存储的状态持久化到本地存储或数据库。

  • 持久化状态插件的示例:

    const localStoragePlugin = store => {store.subscribe((mutation, state) => {// Store the state object as a JSON stringlocalStorage.setItem('my-app-state', JSON.stringify(state));})
    }export default new Vuex.Store({plugins: [localStoragePlugin]
    })
    

    这个插件会在每次变更后把整个 Vuex 状态保存到本地存储。在应用加载时,你需要获取这个状态并初始化 store。

状态管理最佳实践

这里有一些 Vuex 状态管理的通用最佳实践:

  • 保持状态最小化: 仅将需要在多个组件之间共享的数据存储在 store 中。
  • 使用模块: 将您的商店组织成模块,以提高可维护性和可扩展性。
  • 使用命名空间: 使用命名空间来防止命名冲突并改进代码组织。
  • 遵循单向数据流: 操作提交变更,变更更新状态。
  • 使用获取器处理计算属性: 使用获取器从 store 中派生状态,并封装复杂逻辑。
  • 处理动作中的错误: 处理动作中的错误,并提交变更以相应地更新状态。
  • 使用插件扩展功能: 使用插件来扩展 Vuex 的功能。
  • 避免直接修改状态: 永远不要在突变之外直接修改状态。
  • 保持变异同步: 变异应该是同步的,以确保状态变化是可预测的。
  • 记录您的store: 记录您的store,以便其他开发者更容易理解和维护。

相关文章:

vue-20(Vuex 状态管理的最佳实践)

Vuex 状态管理的最佳实践 Vuex 是管理大型 Vue.js 应用状态的一个强大工具&#xff0c;但其有效性取决于其组织和维护的质量。管理不善的 Vuex 存储可能会变得难以控制、难以调试&#xff0c;并成为性能瓶颈。本课程深入探讨构建 Vuex 存储的最佳实践&#xff0c;重点关注可维…...

DAX权威指南8:DAX引擎与存储优化

文章目录 十七、DAX引擎17.1 DAX 引擎的体系结构17.1.1 表格模型的双引擎架构17.1.2 存储引擎的三种模式17.1.2.1 VertiPaq引擎17.1.2.2 DirectQuery 引擎17.1.2.3 对比与最佳实践 17.1.3 数据刷新 17.2 理解 VertiPaq 存储引擎17.2.1 列式数据库17.2.2 VertiPaq 压缩17.2.2.1 …...

智慧货运飞船多维度可视化管控系统

图扑搭建智慧货运飞船可视化系统&#xff0c;借数字孪生技术&#xff0c;高精度复刻货运飞船外观、结构与运行场景。整合多维度数据&#xff0c;实时呈现飞行状态、设备参数等信息&#xff0c;助力直观洞察货运飞船运行逻辑&#xff0c;为航天运维、任务推演及决策提供数字化支…...

电脑开不了机,主板显示67码解决过程

文章目录 现象分析内存条问题BIOS设置问题其它问题 解决清理内存条金手指所需工具操作步骤注意事项 电脑在运行过程中&#xff0c;显示内存不足&#xff0c;重启电脑却无法启动。 现象 System Initialization 主板风扇是转的&#xff0c;也有灯光显示&#xff0c;插上屏幕&am…...

Spring Boot 类加载机制深度解析

Spring Boot 类加载机制深度解析 前言 在 Java 应用开发中&#xff0c;类加载机制是一个重要且复杂的话题。Spring Boot 作为现代 Java 开发的主流框架&#xff0c;其类加载机制更是值得深入了解。本文将从基础概念到实际应用&#xff0c;全面解析 Spring Boot 的类加载机制。…...

Python 训练营打卡 Day 45

TensorBoard 简单来说&#xff0c;TensorBoard 是 TensorFlow 自带的一个「可视化工具」&#xff0c;就像给机器学习模型训练过程装了一个「监控屏幕」。你可以用它直观看到训练过程中的数据变化&#xff08;比如损失值、准确率&#xff09;、模型结构、数据分布等&#xff0c…...

自托管图书搜索引擎Bookologia

简介 什么是 Bookologia &#xff1f; Bookologia 是一个专门的书籍搜索引擎&#xff0c;可以在几秒钟内找到任何书籍。它是开源的&#xff0c;可以轻松自托管在 Docker 上&#xff0c;为用户提供一个简单而高效的书籍查找体验。 主要特点 简洁的用户界面&#xff1a;界面设计…...

前端flex、grid布局

flex布局 弹性布局是指通过调整其内元素的宽高&#xff0c;从而在任何的显示设备上实现对可用显示空间最佳填充的能力。弹性容器扩展其内元素来填充可用空间&#xff0c;或将其收缩来避免溢出 简单来说&#xff0c;弹性盒子模型&#xff0c;是为了你的网页可以在不同分辨率设…...

Maven相关问题:jna版本与ES冲突 + aop失效

文章目录 1、背景2、解决3、一点思考4、环境升级导致AOP失效5、okhttp Bean找不到6、总结 记录一些Maven依赖相关的思考 1、背景 做一个监控指标收集&#xff0c;用一下jna依赖&#xff1a; <dependency><groupId>net.java.dev.jna</groupId><artifact…...

Tomcat全方位监控实施方案指南

#作者&#xff1a;程宏斌 文章目录 一&#xff0e;二进制部署1、安装包信息2、新建配置文件2.1 配置config.yaml文件2.2 上传jar包 3、修改配置3.1 备份3.2 修改bin目录下的startup.sh文件 4、重启tomcat5、访问测试 二&#xff0e;docker部署1、临时方案1.1、重新启动容器1.2…...

开源PHP在线客服系统源码搭建教程

在当今数字化时代&#xff0c;在线客服系统已成为企业与客户沟通的重要桥梁。开源PHP客服系统因其灵活性、低成本和高可定制性而受到众多企业的青睐。本文将介绍几款优秀的开源PHP客服系统&#xff0c;并提供详细的搭建教程。 演示网站&#xff1a;gofly.v1kf.com 1.1 主流开源…...

centos7升级glibic-2.28

centos7升级glibic-2.28 最近使用trae连接服务器的时候&#xff0c;提示远程系统不兼容: Trae CN需要glibc 2.28或更高版本。检测到的版本: 2.17。下面是升级步骤。centos7默认的glibc不支持node v18及以上。 1、进入/home/download目录(没有download&#xff0c;则新建一个)…...

在Docker里面运行Docker

Docker 凭借其轻量级和可移植的容器,无疑改变了软件开发和部署的世界。但如果我告诉你 Docker 本身可以在另一个 Docker 容器中运行,你会怎么想?没错!这个概念通常被称为“Docker Inside Docker”或“DinD”,它为开发人员和系统管理员开辟了一个全新的可能性领域。在这篇博…...

设计模式复习小结

1.容易忘得设计原则 接口隔离&#xff1a;指接口中的功能太杂则可以拆分一下。防止实现类实现了接口后自动依赖了一些不需要的功能。不同功能拆分成不同的接口。 里氏代换&#xff1a;强调父类能出现的地方&#xff0c;子类一定能正常跑。 迪米特法则&#xff1a;又称最少知…...

To be or Not to be, That‘s a Token——论文阅读笔记——Beyond the 80/20 Rule和R2R

本周又在同一方向上刷到两篇文章&#xff0c;可以说&#xff0c;……同学们确实卷啊&#xff0c;要不卷卷开放场域的推理呢&#xff1f; 这两篇都在讲&#xff1a;如何巧妙的利用带有分支能力的token来提高推理性能或效率的。 第一篇叫 Beyond the 80/20 Rule: High-Entropy Mi…...

【基础】每天掌握一个Linux命令 - awk

目录 【基础】每天掌握一个Linux命令 - awk一、工具概述二、安装方式Ubuntu/Debian系统&#xff1a;CentOS/RHEL系统&#xff1a;macOS系统&#xff1a; 三、核心功能四、基础用法基本语法常用选项内置变量基本操作示例1. 打印文件所有内容2. 打印每行的第一个字段3. 指定分隔符…...

《UE5_C++多人TPS完整教程》学习笔记37 ——《P38 变量复制(Variable Replication)》

本文为B站系列教学视频 《UE5_C多人TPS完整教程》 —— 《P38 变量复制&#xff08;Variable Replication&#xff09;》 的学习笔记&#xff0c;该系列教学视频为计算机工程师、程序员、游戏开发者、作家&#xff08;Engineer, Programmer, Game Developer, Author&#xff09…...

AWS API Gateway配置日志

问题 访问API Gateway接口出现了403问题&#xff0c;具体报错如下&#xff1a; {"message":"Missing Authentication Token"}需要配置AWS API Gateway日志&#xff0c;看请求过程是什么样子的。 API Gateway 先找到API Gateway的的日志角色&#xff0c…...

Towards Open World Object Detection概述(论文)

论文&#xff1a;https://arxiv.org/abs/2103.02603 代码&#xff1a;https://github.com/JosephKJ/OWOD Towards Open World Object Detection 迈向开放世界目标检测 Abstract 摘要 Humans have a natural instinct to identify unknown object instances in their environ…...

轻松备份和恢复 Android 系统 | 4 种解决方案

我们通常会在 Android 手机上存储大量重要的个人数据&#xff0c;包括照片、视频、联系人、信息等等。如果您不想丢失宝贵的数据&#xff0c;可以备份 Android 数据。当您需要访问和使用这些数据时&#xff0c;可以将其恢复到 Android 设备。如果您想了解 Android 备份和恢复&a…...

具备强大的数据处理和分析能力的智慧地产开源了

智慧地产视觉监控平台是一款功能强大且简单易用的实时算法视频监控系统。它的愿景是最底层打通各大芯片厂商相互间的壁垒&#xff0c;省去繁琐重复的适配流程&#xff0c;实现芯片、算法、应用的全流程组合&#xff0c;从而大大减少企业级应用约95%的开发成本。 AI是新形势下数…...

RK3588和FPGA桥片之间IO电平信号概率性不能通信原因

1.GPIO管脚配置问题 RK3588对IO进行配置的时候&#xff0c;如果配置为多功能复用&#xff0c;没有明确IO功能&#xff0c;可能引起信号接收不稳定&#xff0c; 需要在驱动中设备树中配置管脚为GPIO功能&#xff0c;确保没有功能复用的干扰。 2.上下拉电阻阻值设置不当 GPIO引脚…...

【iSAQB软件架构】软件架构中构建块的视图:黑箱、灰箱和白箱及其交互机制

在软件架构描述中&#xff0c;黑箱视图&#xff08;Black-box&#xff09;、灰箱视图&#xff08;Gray-box&#xff09;和白箱视图&#xff08;White-box&#xff09; 是不同抽象层级的构建模块表示方式&#xff0c;用于满足不同受众和设计阶段的需求。以下是基于ISAQB标准的清…...

.net jwt实现

.NET 中实现 JWT 认证&#xff1a;详细指南 在现代的 Web 应用开发中&#xff0c;安全认证是至关重要的一环。JSON Web Token&#xff08;JWT&#xff09;作为一种广泛使用的认证机制&#xff0c;为 API 提供了安全、便捷的身份验证方式。本文将详细介绍如何在 ASP.NET Core 项…...

LangChain【7】之工具创建和错误处理策略

文章目录 一 LangChain 自定义工具概述二创建自定义工具的三种方法2.1 方法一&#xff1a;tool 装饰器2.1.1 同步方法案例2.1.2 工具描述方式1&#xff1a;传参2.1.3 工具描述方式2&#xff1a;文档字符串 2.2 方法二&#xff1a;StructuredTool类2.2.1 StructuredTool创建自定…...

如何在电脑上轻松访问 iPhone 文件

我需要将 iPhone 下载文件夹中的文件传输到 Windows 11 电脑上。我该怎么做&#xff1f;我可以在 Windows 11 上访问 iPhone 下载吗&#xff1f; 由于 iOS 和 Windows 系统之间的差异&#xff0c;在 PC 上访问 iPhone 文件似乎颇具挑战性。然而&#xff0c;只要使用正确的工具…...

Eureka REST 相关接口

可供非 Java 应用程序使用的 Eureka REST 操作。 appID 是应用程序的名称&#xff0c;instanceID 是与实例关联的唯一标识符。在 AWS 云中&#xff0c;instanceID 是实例的实例 ID&#xff1b;在其他数据中心&#xff0c;它是实例的主机名。 对于 XML/JSON&#xff0c;HTTP 的…...

C语言字符数组输入输出方法大全(附带实例)

在 C语言中&#xff0c;字符数组是一种特殊的数组&#xff0c;用于存储和处理字符串。理解字符数组的输入和输出操作对于初学者来说至关重要&#xff0c;因为这是处理文本数据的基础。 字符数组的定义与初始化 在讨论输入输出之前&#xff0c;我们先来回顾一下字符数组的定义…...

短视频矩阵SaaS系统:开源部署与核心功能架构指南

一、系统架构概述 短视频矩阵系统是基于SaaS&#xff08;软件即服务&#xff09;模式的多平台内容管理解决方案&#xff0c;通过开源技术实现账号聚合、智能创作、跨平台分发及数据闭环。系统采用微服务架构&#xff0c;支持高并发场景下的弹性扩展。 二、核心功能模块开发逻辑…...

每日算法 -【Swift 算法】电话号码字母组合

&#x1f680; LeetCode 字符串数字映射&#xff08;Swift&#xff09;——电话号码字母组合 在日常刷题或面试中&#xff0c;我们经常会遇到字符串 回溯组合的问题。这道经典题——电话号码的字母组合 就是典型代表。本文将带你用 Swift 实现这道题&#xff0c;思路清晰&…...