Vue学习之--------深入理解Vuex之模块化编码(2022/9/4)

简介: 这篇文章详细介绍了Vuex的模块化编码和命名空间的使用,旨在让代码更易于维护并提高数据分类的明确性。内容包括模块化和命名空间的概念、如何在store中配置模块、以及如何在组件中使用模块化的数据。文章通过实战项目案例,展示了如何拆分`store/index.js`文件,创建`count.js`和`person.js`模块,并在`Count.vue`和`Person.vue`组件中使用这些模块。最后,文章还提供了测试效果和一些使用注意点。

在以下文章的基础上
1、深入理解Vuex、原理详解、实战应用:https://blog.csdn.net/weixin_43304253/article/details/126651368
2、深入理解Vuex之getters、mapState、mapGetters:https://blog.csdn.net/weixin_43304253/article/details/126679366
3、深入理解Vuex之多组件共享数据:https://blog.csdn.net/weixin_43304253/article/details/126685612

文章目录

  • 1、模块化+命名空间
  • 2、项目中的实战应用
    • 2.1 项目结构
    • 2.1 store下的文件配置
      • 2.1.1 count.js(专门管理求和)
      • 2.1.2 person.js(专门管理人员)
      • 2.1.3 index.j(进行汇总,统一对外暴露)
    • 2.2 components下的组件
      • 2.2.1 Count.vue
      • 2.2.2 Person.vue
    • 2.3 App.vue
    • 2.4 main.js
  • 3、测试效果
  • 4、小范围的拆分store/index.js
  • 5、注意点

组件化编码的好处、方便后期的维护、减少系统的耦合性

1、模块化+命名空间

  1. 目的:让代码更好维护,让多种数据分类更加明确。

  2. 修改store.js

   const countAbout = {
     namespaced:true,//开启命名空间
     state:{x:1},
     mutations: { ... },
     actions: { ... },
     getters: {
       bigSum(state){
          return state.sum * 10
       }
     }
   }

   const personAbout = {
     namespaced:true,//开启命名空间
     state:{ ... },
     mutations: { ... },
     actions: { ... }
   }

   const store = new Vuex.Store({
     modules: {
       countAbout,
       personAbout
     }
   })
  1. 开启命名空间后,组件中读取state数据:
   //方式一:自己直接读取
   this.$store.state.personAbout.list
   //方式二:借助mapState读取:
   ...mapState('countAbout',['sum','school','subject']),
  1. 开启命名空间后,组件中读取getters数据:
   //方式一:自己直接读取
   this.$store.getters['personAbout/firstPersonName']
   //方式二:借助mapGetters读取:
   ...mapGetters('countAbout',['bigSum'])
  1. 开启命名空间后,组件中调用dispatch
   //方式一:自己直接dispatch
   this.$store.dispatch('personAbout/addPersonWang',person)
   //方式二:借助mapActions:
   ...mapActions('countAbout',{incrementOdd:'jiaOdd',incrementWait:'jiaWait'})
  1. 开启命名空间后,组件中调用commit
   //方式一:自己直接commit
   this.$store.commit('personAbout/ADD_PERSON',person)
   //方式二:借助mapMutations:
   ...mapMutations('countAbout',{increment:'JIA',decrement:'JIAN'}),

2、项目中的实战应用

2.1 项目结构

在这里插入图片描述

2.1 store下的文件配置

2.1.1 count.js(专门管理求和)

//求和相关配置
export default {
    namespaced: true,
    actions: {
        jia(context, value) {
            context.commit('JIA', value)
        },

        jiaOdd(context, value) {
            if (context.state.sum % 2) {
                context.commit('JIA', value)
            }
        },

        jiaWait(context, value) {
            setTimeout(() => {
                context.commit('JIA', value)
            }, 500);
        }
    },
    mutations: {
        JIA(state, value) {
            state.sum += value
        },
        JIAN(state, value) {
            state.sum -= value
        }

    },
    state: {
        sum: 0,//当前的和,
        name: '张三',
        address: "广州"
    },
    getters: {
        bigSum(state) {
            return state.sum * 10
        }
    }

}

2.1.2 person.js(专门管理人员)

在action中进行业务处理时,使用axios调用接口。axios的安装使用:npm i axios

//人员管理相关配置
//引入axios 
import axios from 'axios'
import { nanoid } from 'nanoid'

export default {
    namespaced: true,
    actions: {
        addPersonZheng(context, value) {
            if (value.name.indexOf('郑') === 0) {
                console.log("123")
                context.commit('ADD_PERSON', value)
            } else {
                alert("添加的人必须性郑")
            }
        },

        addPersonServer(context) {
            axios.get('https://api.uixsj.cn/hitokoto/get?type=social').then(
                response => {
                    context.commit('ADD_PERSON', { id: nanoid(), name: response.data })
                },
                error => {
                    alert(error.message)
                }
            )

        }

    },

    mutations: {
        ADD_PERSON(state, value) {
            state.personList.unshift(value)
        }

    },
    state: {
        personList: [{ id: 'A001', name: '张三' }]
    },
    getters: {
        firstPersonName(state) {
            return state.personList[0].name
        }
    }

}

2.1.3 index.j(进行汇总,统一对外暴露)

//该文件用于创建Vuex中最为核心的store

//引入Vuex
import Vuex from 'vuex'
//引入Vue
import Vue from 'vue'

import countOptions from './count'
import personOptions from './person'

//使用插件
Vue.use(Vuex)

//第一种形式
//创建并且暴露store
export default new Vuex.Store({
    modules: {
        countAbout: countOptions,
        personAbout: personOptions
    }

})

//第二种形式

// //创建store
// const store = new Vuex.Store({
//     actions,
//     mutations,
//     state,
// })

// //导出store
// export default store

2.2 components下的组件

2.2.1 Count.vue

<template>
  <div>
    <h1>当前求和为:{
  
  { sum }}</h1>
    <h2>当前求和扩大十倍为:{
  
  { bigSum }}</h2>
    <h3>你好,{
  
  { name }}在{
  
  { address }}工作</h3>
    <select v-model.number="n">
      <option value="1">1</option>
      <option value="2">2</option>
      <option value="3">3</option>
    </select>
    <button @click="increment(n)">+</button>
    <button @click="decrement(n)">-</button>
    <button @click="incrementOdd(n)">当前求和为奇数再加</button>
    <button @click="incrementWait(n)">等一等再加</button>
    <hr>
    <h3 style="color:pink">当前用户数量:{
  
  {personList.length}}</h3>
  </div>
</template>

<script>
import { mapState, mapGetters, mapActions, mapMutations } from "vuex";
export default {
  name: "Count",
  data() {
    return {
      n: 1, //用户选择的数字
    };
  },
  methods: {
    //借助mapMutations生成对应的方法,方法中会调用commit去联系mutations(对象写法)
    ...mapMutations('countAbout',{ increment: "JIA", decrement: "JIAN" }),

    //借助mapActions生成对应的方法,方法中会调用dispatch去联系actions(对象写法)
    ...mapActions('countAbout',{ incrementOdd: "jiaOdd", incrementWait: "jiaWait" }),
  },
  computed: {
    //数组写法
    ...mapState('countAbout',["sum", "name", "address"]),
    ...mapState('personAbout',['personList']),

    //数组写法
    ...mapGetters('countAbout',["bigSum"]),
  },
};
</script>

<style lang="css">
button {
  margin-left: 5px;
}
</style>

2.2.2 Person.vue

<template>
  <div>
    <h1>人员信息展示</h1>
    <h3 style="color: pink">Count组件的和为:{
  
  { Count }}</h3>
    <h3>列表中第一个人的名字是:{
  
  { firstPersonName }}</h3>
    <input type="text" placeholder="请输入姓名" v-model="name" />
    <button @click="add">添加</button>
    <button @click="addZheng">添加一个姓郑的人</button>
    <button @click="addPersonServer">添加一个人,名字随机</button>
    <ul>
      <li v-for="p in personList" :key="p.id">{
  
  { p.name }}</li>
    </ul>
  </div>
</template>

<script>
import { nanoid } from "nanoid";
import { mapState, mapGetters, mapActions, mapMutations } from "vuex";
export default {
  name: "Person",
  data() {
    return {
      name: "",
      n: 1,
    };
  },
  methods: {
    add() {
      const personObj = { id: nanoid(), name: this.name };
      this.$store.commit("personAbout/ADD_PERSON", personObj);
      this.name = "";
    },

    addZheng() {
      const personObj = { id: nanoid(), name: this.name };
      this.$store.dispatch("personAbout/addPersonZheng", personObj);
      this.name = "";
    },

    addPersonServer() {
      const personObj = { id: nanoid(), name: this.name };
      this.$store.dispatch("personAbout/addPersonServer", personObj);
      this.name = "";
    },
  },
  computed: {
    personList() {
      return this.$store.state.personAbout.personList;
    },
    Count() {
      return this.$store.state.countAbout.sum;
    },

    firstPersonName() {
      return this.$store.getters["personAbout/firstPersonName"];
    },
  },
};
</script>

<style lang="css">
</style>

2.3 App.vue

<template>
    <div>
        <Count/>
        <hr>
        <Person/>
    </div>
</template>

<script>
    import Count from './components/Count'
    import Person from './components/Person.vue'
    export default {
        name:'App',
        components:{Count,Person},
    }
</script>

2.4 main.js

// The Vue build version to load with the `import` command
// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
import Vue from 'vue'
import App from './App'
import router from './router'

//引入store
import store from './store/index.js'

Vue.config.productionTip = false

/* eslint-disable no-new */
new Vue({
  el: '#app',
  router,
  store,
  render: h => h(App),
  beforenCreate() {
    Vue.prototype.$bus = this
  }

})

3、测试效果

在这里插入图片描述
在这里插入图片描述

4、小范围的拆分store/index.js

//该文件用于创建Vuex中最为核心的store

//引入Vuex
import Vuex from 'vuex'
//引入Vue
import Vue from 'vue'

//引入axios 
import axios from 'axios'

import { nanoid } from 'nanoid'
//使用插件
Vue.use(Vuex)

const countOptions = {
    namespaced: true,
    //准备action-- 用于响应组件中的动作
    actions: {
        jia(context, value) {
            context.commit('JIA', value)
        },

        jiaOdd(context, value) {
            if (context.state.sum % 2) {
                context.commit('JIA', value)
            }
        },

        jiaWait(context, value) {
            setTimeout(() => {
                context.commit('JIA', value)
            }, 500);
        }
    },
    //准备mutations-- 用于操作数据(state)
    mutations: {
        JIA(state, value) {
            state.sum += value
        },
        JIAN(state, value) {
            state.sum -= value
        }

    },
    //准备state--用于存储数据
    state: {
        sum: 0,//当前的和,
        name: '张三',
        address: "广州"
    },
    //准备getters
    getters: {
        bigSum(state) {
            return state.sum * 10
        }
    }

}

const personOptions = {
    namespaced: true,
    actions: {
        addPersonZheng(context, value) {
            if (value.name.indexOf('郑') === 0) {
                console.log("123")
                context.commit('ADD_PERSON', value)
            } else {
                alert("添加的人必须性郑")
            }
        },

        addPersonServer(context) {
            axios.get('https://api.uixsj.cn/hitokoto/get?type=social').then(
                response => {
                    context.commit('ADD_PERSON', { id: nanoid(), name: response.data })
                },
                error => {
                    alert(error.message)
                }
            )

        }

    },

    mutations: {
        ADD_PERSON(state, value) {
            state.personList.unshift(value)
        }

    },
    state: {
        personList: [{ id: 'A001', name: '张三' }]
    },
    getters: {
        firstPersonName(state) {
            return state.personList[0].name
        }
    }

}

//第一种形式
//创建并且暴露store
export default new Vuex.Store({
    modules: {
        countAbout: countOptions,
        personAbout: personOptions
    }

})

//第二种形式

// //创建store
// const store = new Vuex.Store({
//     actions,
//     mutations,
//     state,
// })

// //导出store
// export default store

5、注意点

不会这个、找不到你的这个”组件化“
在这里插入图片描述

相关文章
|
8天前
|
JavaScript
vue使用iconfont图标
vue使用iconfont图标
56 1
|
18天前
|
JavaScript 关系型数据库 MySQL
基于VUE的校园二手交易平台系统设计与实现毕业设计论文模板
基于Vue的校园二手交易平台是一款专为校园用户设计的在线交易系统,提供简洁高效、安全可靠的二手商品买卖环境。平台利用Vue框架的响应式数据绑定和组件化特性,实现用户友好的界面,方便商品浏览、发布与管理。该系统采用Node.js、MySQL及B/S架构,确保稳定性和多功能模块设计,涵盖管理员和用户功能模块,促进物品循环使用,降低开销,提升环保意识,助力绿色校园文化建设。
|
2月前
|
JavaScript API 开发者
Vue是如何进行组件化的
Vue是如何进行组件化的
|
2月前
|
JavaScript 前端开发 开发者
Vue是如何进行组件化的
Vue是如何进行组件化的
|
2月前
|
存储 JavaScript 前端开发
介绍一下Vue的核心功能
介绍一下Vue的核心功能
|
JavaScript 前端开发
一周精通Vue(二)组件访问、插槽、组件作用域、模块化
vue、组件访问、插槽、组件作用域、模块化、vue基础、快速上手vue、vue详细讲解
16382 0
|
2月前
|
JavaScript 前端开发 开发者
vue学习第一章
欢迎来到我的博客!我是瑞雨溪,一名热爱前端的大一学生,专注于JavaScript与Vue,正向全栈进发。博客分享Vue学习心得、命令式与声明式编程对比、列表展示及计数器案例等。关注我,持续更新中!🎉🎉🎉
49 1
vue学习第一章
|
2月前
|
JavaScript 前端开发 索引
vue学习第三章
欢迎来到瑞雨溪的博客,一名热爱JavaScript与Vue的大一学生。本文介绍了Vue中的v-bind指令,包括基本使用、动态绑定class及style等,希望能为你的前端学习之路提供帮助。持续关注,更多精彩内容即将呈现!🎉🎉🎉
34 1
|
2月前
|
缓存 JavaScript 前端开发
vue学习第四章
欢迎来到我的博客!我是瑞雨溪,一名热爱JavaScript与Vue的大一学生。本文介绍了Vue中计算属性的基本与复杂使用、setter/getter、与methods的对比及与侦听器的总结。如果你觉得有用,请关注我,将持续更新更多优质内容!🎉🎉🎉
41 1
vue学习第四章
|
2月前
|
JavaScript 前端开发 算法
vue学习第7章(循环)
欢迎来到瑞雨溪的博客,一名热爱JavaScript和Vue的大一学生。本文介绍了Vue中的v-for指令,包括遍历数组和对象、使用key以及数组的响应式方法等内容,并附有综合练习实例。关注我,将持续更新更多优质文章!🎉🎉🎉
32 1
vue学习第7章(循环)