二次封装 Vuex for Uniapp 微信小程序开发

news2025/6/2 16:31:56

作为高级前端开发工程师,我将为你提供一个针对 Uniapp + Vue2 + Vuex 的 Store 二次封装方案,使团队成员能够更便捷地使用和管理状态。

封装目标

  1. 模块化管理状态

  2. 简化调用方式

  3. 提供类型提示(在 Vue2 中尽可能实现)

  4. 便于维护和查找

封装方案

1. 目录结构重构

首先,我们重构 store 的目录结构:

store/
├── index.js              # 主入口文件
├── modules/              # 模块目录
│   ├── user.js           # 用户模块
│   ├── cart.js           # 购物车模块
│   └── ...               # 其他模块
└── store-utils.js        # 封装的工具方法

 2. 主入口文件封装 (store/index.js)

import Vue from 'vue'
import Vuex from 'vuex'
import * as utils from './store-utils'

Vue.use(Vuex)

// 自动加载 modules 目录下的所有模块
const modules = utils.autoLoadModules()

const store = new Vuex.Store({
  modules,
  // 全局配置
  strict: process.env.NODE_ENV !== 'production'
})

// 挂载工具方法到 Vue 原型
Vue.prototype.$storeUtils = utils

export default store

3. 工具方法封装 (store/store-utils.js)

/**
 * 自动加载 modules 目录下的所有模块
 */
export function autoLoadModules() {
  const modules = {}
  const requireModule = require.context(
    // 搜索目录
    './modules',
    // 是否搜索子目录
    false,
    // 匹配文件的正则
    /\.js$/
  )
  
  requireModule.keys().forEach(fileName => {
    // 获取模块名(去掉文件扩展名)
    const moduleName = fileName.replace(/(\.\/|\.js)/g, '')
    modules[moduleName] = {
      namespaced: true,
      ...requireModule(fileName).default
    }
  })
  
  return modules
}

/**
 * 简化 mapState 调用
 * @param {string} moduleName 模块名
 * @param {Array|Object} states 状态名数组或对象
 */
export function mapState(moduleName, states) {
  if (!moduleName) return {}
  
  if (Array.isArray(states)) {
    const result = {}
    states.forEach(key => {
      result[key] = function() {
        return this.$store.state[moduleName][key]
      }
    })
    return result
  }
  
  // 对象形式
  const result = {}
  Object.keys(states).forEach(key => {
    result[key] = function() {
      return this.$store.state[moduleName][states[key]]
    }
  })
  return result
}

/**
 * 简化 mapGetters 调用
 * @param {string} moduleName 模块名
 * @param {Array|Object} getters getter名数组或对象
 */
export function mapGetters(moduleName, getters) {
  if (!moduleName) return {}
  
  const namespace = moduleName + '/'
  
  if (Array.isArray(getters)) {
    const result = {}
    getters.forEach(getterName => {
      result[getterName] = function() {
        return this.$store.getters[namespace + getterName]
      }
    })
    return result
  }
  
  // 对象形式
  const result = {}
  Object.keys(getters).forEach(key => {
    result[key] = function() {
      return this.$store.getters[namespace + getters[key]]
    }
  })
  return result
}

/**
 * 简化 mapMutations 调用
 * @param {string} moduleName 模块名
 * @param {Array|Object} mutations mutation名数组或对象
 */
export function mapMutations(moduleName, mutations) {
  if (!moduleName) return {}
  
  const namespace = moduleName + '/'
  
  if (Array.isArray(mutations)) {
    const result = {}
    mutations.forEach(mutationName => {
      result[mutationName] = function(payload) {
        this.$store.commit(namespace + mutationName, payload)
      }
    })
    return result
  }
  
  // 对象形式
  const result = {}
  Object.keys(mutations).forEach(key => {
    result[key] = function(payload) {
      this.$store.commit(namespace + mutations[key], payload)
    }
  })
  return result
}

/**
 * 简化 mapActions 调用
 * @param {string} moduleName 模块名
 * @param {Array|Object} actions action名数组或对象
 */
export function mapActions(moduleName, actions) {
  if (!moduleName) return {}
  
  const namespace = moduleName + '/'
  
  if (Array.isArray(actions)) {
    const result = {}
    actions.forEach(actionName => {
      result[actionName] = function(payload) {
        return this.$store.dispatch(namespace + actionName, payload)
      }
    })
    return result
  }
  
  // 对象形式
  const result = {}
  Object.keys(actions).forEach(key => {
    result[key] = function(payload) {
      return this.$store.dispatch(namespace + actions[key], payload)
    }
  })
  return result
}

/**
 * 创建模块的快捷访问方式
 * @param {string} moduleName 模块名
 */
export function createModuleHelpers(moduleName) {
  return {
    state: (stateName) => {
      return this.$store.state[moduleName][stateName]
    },
    getter: (getterName) => {
      return this.$store.getters[`${moduleName}/${getterName}`]
    },
    commit: (mutationName, payload) => {
      this.$store.commit(`${moduleName}/${mutationName}`, payload)
    },
    dispatch: (actionName, payload) => {
      return this.$store.dispatch(`${moduleName}/${actionName}`, payload)
    }
  }
}

4. 模块示例 (store/modules/user.js)

const state = {
  token: '',
  userInfo: null,
  isLogin: false
}

const getters = {
  getUserName: state => {
    return state.userInfo?.name || ''
  },
  getUserId: state => {
    return state.userInfo?.id || ''
  }
}

const mutations = {
  SET_TOKEN(state, token) {
    state.token = token
  },
  SET_USER_INFO(state, userInfo) {
    state.userInfo = userInfo
    state.isLogin = !!userInfo
  },
  CLEAR_USER(state) {
    state.token = ''
    state.userInfo = null
    state.isLogin = false
  }
}

const actions = {
  login({ commit }, { username, password }) {
    return new Promise((resolve, reject) => {
      // 模拟登录请求
      uni.request({
        url: '/api/login',
        method: 'POST',
        data: { username, password },
        success: (res) => {
          commit('SET_TOKEN', res.data.token)
          commit('SET_USER_INFO', res.data.userInfo)
          resolve(res.data)
        },
        fail: (err) => {
          reject(err)
        }
      })
    })
  },
  logout({ commit }) {
    commit('CLEAR_USER')
    return Promise.resolve()
  }
}

export default {
  state,
  getters,
  mutations,
  actions
}

使用方式

1. 组件中使用

import { mapState, mapGetters, mapMutations, mapActions } from '@/store/store-utils'

export default {
  computed: {
    // 传统方式
    ...mapState('user', ['token', 'userInfo', 'isLogin']),
    ...mapGetters('user', ['getUserName', 'getUserId']),
    
    // 或者使用对象形式
    ...mapState('user', {
      myToken: 'token',
      myUserInfo: 'userInfo'
    })
  },
  methods: {
    ...mapMutations('user', ['SET_USER_INFO']),
    ...mapActions('user', ['login', 'logout']),
    
    // 示例方法
    handleLogin() {
      this.login({ username: 'test', password: '123456' })
        .then(() => {
          uni.showToast({ title: '登录成功' })
        })
        .catch(err => {
          uni.showToast({ title: '登录失败', icon: 'none' })
        })
    }
  }
}

2. 快捷访问方式

export default {
  methods: {
    someMethod() {
      // 使用快捷访问
      const userHelpers = this.$storeUtils.createModuleHelpers('user')
      
      // 获取状态
      const token = userHelpers.state('token')
      
      // 获取getter
      const userName = userHelpers.getter('getUserName')
      
      // 提交mutation
      userHelpers.commit('SET_USER_INFO', { name: '张三' })
      
      // 分发action
      userHelpers.dispatch('logout').then(() => {
        console.log('登出成功')
      })
    }
  }
}

3. 在JS文件中使用

import store from '@/store'

// 直接使用store
const token = store.state.user.token

// 提交mutation
store.commit('user/SET_TOKEN', 'new-token')

// 分发action
store.dispatch('user/login', { username: 'test', password: '123456' })
  .then(() => {
    console.log('登录成功')
  })

最佳实践建议

  1. 命名规范

    • 模块名使用小驼峰命名法 (user, shoppingCart)

    • state 属性使用小驼峰命名法

    • mutations 使用大写蛇形命名法 (SET_USER_INFO)

    • actions 使用小驼峰命名法

  2. 文档注释
    在每个模块文件顶部添加注释说明模块用途,重要的 state、getter、mutation 和 action 添加注释

  3. 模块拆分

    • 按业务功能拆分模块

    • 避免单个模块过大

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2393906.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

详细到用手撕transformer下半部分

之前我们讨论了如何实现 Transformer 的核心多头注意力机制,那么这期我们来完整地实现整个 Transformer 的编码器和解码器。 Transformer 架构最初由 Vaswani 等人在 2017 年的论文《Attention Is All You Need》中提出,专为序列到序列(seq2s…

【Sqoop基础】Sqoop生态集成:与HDFS、Hive、HBase等组件的协同关系深度解析

目录 1 Sqoop概述与大数据生态定位 2 Sqoop与HDFS的深度集成 2.1 技术实现原理 2.2 详细工作流程 2.3 性能优化实践 3 Sqoop与Hive的高效协同 3.1 集成架构设计 3.2 数据类型映射处理 3.3 案例演示 4 Sqoop与HBase的实时集成 4.1 数据模型转换挑战 4.2 详细集成流程…

MySQL + CloudCanal + Iceberg + StarRocks 构建全栈数据服务

简述 在业务数据快速膨胀的今天,企业对 低成本存储 与 实时查询分析能力 的需求愈发迫切。 本文将带你实战构建一条 MySQL 到 Iceberg 的数据链路,借助 CloudCanal 快速完成数据迁移与同步,并使用 StarRocks 完成数据查询等操作&#xff0c…

截屏精灵:轻松截屏,高效编辑

在移动互联网时代,截图已经成为我们日常使用手机时的一项基本操作。无论是记录重要信息、分享有趣内容,还是进行学习和工作,一款好用的截图工具都能极大地提升我们的效率。截屏精灵就是这样一款功能强大、操作简单的截图工具,它不…

【JavaWeb】基本概念、web服务器、Tomcat、HTTP协议

目录 1. 基本概念1.1 基本概念1.2 web应用程序1.3 静态web1.4 动态web 2. web服务器3. tomcat详解3.1 安装3.2 启动3.3 配置3.3.1 配置启动的端口号3.3.2 配置主机的名称3.3.3 其他常用配置项日志配置数据源配置安全配置 3.4 发布一个网站 4. Http协议4.1 什么是http4.2 http的…

云计算Linux Rocky day02(安装Linux系统、设备表示方式、Linux基本操作)

云计算Linux Rocky day02(安装Linux系统、设备表示方式、Linux基本操作) 目录 云计算Linux Rocky day02(安装Linux系统、设备表示方式、Linux基本操作)1、虚拟机VMware安装Rocky2、Linux命令行3、Linux Rocky修改字体大小和背景颜…

在 ODROID-H3+ 上安装 Win11 系统

在 ODROID-H3 上安装 Windows 11 系统。 以下是完整的步骤,包括 BIOS 设置、U 盘制作、安装和驱动处理,全程不保留之前的系统数据。 ✅ 准备工作 1. 准备一个 ≥8GB 的 USB 启动盘 用另一台电脑制作 Windows 11 安装盘。 👉 推荐工具&…

使用el-input数字校验,输入汉字之后校验取消不掉

先说说复现方式 本来input是只能输入数字的&#xff0c;然后你不小心输入了汉字&#xff0c;触发校验了&#xff0c;然后这时候&#xff0c;你发现校验取消不掉了 就这样了 咋办啊&#xff0c;你一看校验没错啊&#xff0c;各种number啥的也写了,发现没问题啊 <el-inputv…

Docker容器启动失败的常见原因分析

我们在开发部署的时候&#xff0c;用 Docker 打包环境&#xff0c;理论上是“我装好了你就能跑”。但理想很丰满&#xff0c;现实往往一 docker run 下去就翻车了。 今天来盘点一下我实际工作中经常遇到的 Docker 容器启动失败的常见原因&#xff0c;顺便给点 debug 的小技巧&a…

立志成为一名优秀测试开发工程师(第七天)——unittest框架的学习

目录 unittest框架的学习 一、测试类的编写 创建相关测试类cal.py、CountTest.py 二、常见断言方法 使用unittest单元测试框架编写测试用例CountTest.py 注意&#xff1a;执行的时候光标一定要放在括号后面&#xff0c;鼠标右键运行 三、对测试环境的初始化和清除模块…

论坛系统(4)

用户详情 获取用户信息 实现逻辑 ⽤⼾提交请求&#xff0c;服务器根据是否传⼊Id参数决定返回哪个⽤⼾的详情 1. 不传⽤⼾Id&#xff0c;返回当前登录⽤⼾的详情(从session获取) 2. 传⼊⽤⼾Id&#xff0c;返回指定Id的⽤⼾详情(根据用户id去查) 俩种方式获得用户信息 参…

力扣面试150题--二叉树的层平均值

Day 54 题目描述 思路 初次做法&#xff08;笨&#xff09;&#xff1a;使用两个队列&#xff0c;一个队列存放树的节点&#xff0c;一个队列存放对应节点的高度&#xff0c;使用x存放上一个节点&#xff0c;highb存放上一个节点的高度&#xff0c;sum存放当前层的节点值之和…

【Doris入门】Doris初识:分布式分析型数据库的核心价值与架构解析

目录 1 Doris简介与核心价值 2 Doris架构深度解析 2.1 Frontend&#xff08;FE&#xff09;架构 2.2 Backend&#xff08;BE&#xff09;架构 3 Doris核心概念详解 3.1 数据分布模型 3.2 Tablet与Replica 3.3 数据模型 4 Doris关键技术解析 4.1 存储引擎 4.2 查询执…

数据结构与算法学习笔记(Acwing 提高课)----动态规划·区间DP

数据结构与算法学习笔记----动态规划区间DP author: 明月清了个风 first publish time: 2025.5.26 ps⭐️区间DP的特征在于子结构一般是一个子区间上的问题&#xff0c;涉及到的问题也非常多&#xff0c;如环形区间&#xff0c;记录方案数&#xff0c;高精度&#xff0c;二维…

从0到1搭建AI绘画模型:Stable Diffusion微调全流程避坑指南

从0到1搭建AI绘画模型&#xff1a;Stable Diffusion微调全流程避坑指南 系统化学习人工智能网站&#xff08;收藏&#xff09;&#xff1a;https://www.captainbed.cn/flu 文章目录 从0到1搭建AI绘画模型&#xff1a;Stable Diffusion微调全流程避坑指南摘要引言一、数据集构…

从一到无穷大 #46:探讨时序数据库Deduplicate与Compaction的设计权衡

本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。 本作品 (李兆龙 博文, 由 李兆龙 创作)&#xff0c;由 李兆龙 确认&#xff0c;转载请注明版权。 文章目录 引言Compaction AlgorithmsCompact Execution Flow Based On VeloxLocalMergeSource的…

vue3 导出excel

需求&#xff1a;导出自带格式的excel表格 1.自定义二维数组格式 导出 全部代码&#xff1a; <el-button click"exportExcel">导出</el-button> const exportExcel () > {const data [[商品, 单价, 数量, 总价],[A, 100, 1.55, { t: n, f: B2*C2…

day024-网络基础-TCP与UDP、DNS

文章目录 1. 李导推荐书籍2. OSI七层模型2.1 传输层2.2 网络层2.2.1 问&#xff1a;两端处于不同局域网的设备怎么网络通信&#xff1f; 2.3 数据链路层2.4 物理层2.5 图解OSI七层模型 3. 数据传输模式3.1 全双工3.2 半双工3.3 单工 4. TCP 3次握手4.1 抓包 5. TCP 4次挥手5.1 …

专场回顾 | 重新定义交互,智能硬件的未来设计

自2022年起&#xff0c;中国智能硬件行业呈现出蓬勃发展的态势&#xff0c;市场规模不断扩大。一个多月前&#xff0c;“小智AI”在短视频平台的爆火将智能硬件带向了大众视野&#xff0c;也意味着智能硬件已不再仅仅停留在概念和技术层面&#xff0c;而是加速迈向实际落地应用…

WPS 免登录解锁编辑

遇到 WPS 需要登录才能启用编辑功能&#xff1f; 如何免登录使用编辑功能&#xff1f; 方法一 解锁方法 1、关闭 WPS&#xff1b; 2、桌面右键→ “新建”→“文本文档”&#xff0c;粘贴以下内容&#xff08;见最下面&#xff09;&#xff1b;编码保持默认&#xff08;ANSI …