2023/03/09 - Vue学习笔记 - 【组件通信】 vuex的使用和讲解—$store

news2025/1/18 12:44:23

官网:https://vuex.vuejs.org/zh/

1 Vuex存在的意义

概念:专门在 Vue 中实现集中式状态(数据)管理的一个 Vue 插件,对 vue 应用中多个组件的共享状态进行集中式的管理(读/写),也是一种组件间通信的方式,且适用于任意组件间通信。

首先,我们针对于全局事件总线做一个差评:
在这里插入图片描述
在这里插入图片描述
全局事件总线的读写太过混乱了!!!!引出vuex:
将 多个组件所要操作的数据共享出去:
在这里插入图片描述
使用vuex的场景有两种:

  • 1 多个组件依赖于同一状态【数据】
  • 2 来自不同组件的行为需要变更同一状态【数据】

2 引入一个求和例子【base】

在这里插入图片描述
源码如下:

<template>
  <div id="app">
    <Count/>
  </div>
</template>
<script>

import Count from "@/components/Count";

export default {
  name: `App`,
  data() {
    return {
    }
  },
  components: {Count},
  methods: {},
  mounted() {
  }
}
</script>

<style>
</style>

<template>
  <div>
    <h1>当前求和为: {{ sum }}</h1>
    <select v-model.number="number">
      <option :value="1">1</option>
      <option :value="2">2</option>
      <option :value="3">3</option>
    </select>
    <button @click="increment">+</button>
    <button @click="decrement">-</button>
    <button @click="incrementOdd">当前求和为奇数再加</button>
    <button @click="incrementWait">等一等再加</button>
  </div>
</template>
<script>
export default {
  name: `Count`,
  data() {
    return {
      sum: 0,
      number: 1
    }
  },
  methods: {
    increment() {
      this.sum += this.number
    },
    decrement() {
      this.sum -= this.number
    },
    incrementOdd() {
      if (this.number % 2) {
        this.sum += this.number
      }
    },
    incrementWait() {
      setTimeout(() => {
        this.sum += this.number
      }, 1000)
    },
  }
}
</script>

<style>
button {
  margin-left: 10px;
}
</style>

在这里插入图片描述

3 Vuex的架构图

在这里插入图片描述

3.1 State 【存储状态数据】

在这里插入图片描述
官方解读:Vuex 使用单一状态树——用一个对象就包含了全部的应用层级状态。至此它便作为一个“唯一数据源 (SSOT)”而存在。这也意味着,每个应用将仅仅包含一个 store 实例。单一状态树让我们能够直接地定位任一特定的状态片段,在调试的过程中也能轻易地取得整个当前应用状态的快照。

3.2 Mutation 【数据操作】

更改 Vuex 的 store 中的状态的唯一方法是提交 mutation。Vuex 中的 mutation 非常类似于事件:每个 mutation 都有一个字符串的事件类型 (type)和一个回调函数 (handler)。这个回调函数就是我们实际进行状态更改的地方,并且它会接受 state 作为第一个参数,传入的值作为第二个参数。在这里插入图片描述

3.3 Action 【类似于服务员的操作,响应数据】

Action 类似于 mutation,不同在于:Action 提交的是 mutation,而不是直接变更状态。Action 可以包含任意异步操作。
在这里插入图片描述

4 Vuex的引用使用

  • 1 vue2 安装vuex
npm i vuex@3
  • 2 在main.js中引用使用
import Vue from 'vue'
import App from './App.vue'

Vue.config.productionTip = false
import Vuex from 'vuex'
Vue.use(Vuex)

let vm = new Vue({
    render: h => h(App),
    store: 'hello'
}).$mount('#app')

在这里插入图片描述

  • 3 创建store - 涉及到一个import的执行顺序的问题

在这里插入图片描述
源码如下所示:

import Vue from 'vue'
import App from './App.vue'

Vue.config.productionTip = false



// 引入store
import store from "./store/index";

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

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

// 引入Vue
import Vue from "vue";
// 引入vuex
import Vuex from 'vuex'
Vue.use(Vuex)

// actions - 用于响应组件中的动作
const actions = {}

// mutations - 用于操作数据(state)
const mutations = {}

// state - 用于存储数据
const state = {}

// 创建store 暴露store
export default new Vuex.Store({
    actions,
    mutations,
    state
});

在这里插入图片描述

5 vuex 改造求和案例【base】

5.1 严格按照三步走的策略

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

源码如下所示:
main.js

import Vue from 'vue'
import App from './App.vue'

Vue.config.productionTip = false

// 引入store
import store from "./store/index";

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

index.js

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

// 引入Vue
import Vue from "vue";
// 引入vuex
import Vuex from 'vuex'
Vue.use(Vuex)

// actions - 用于响应组件中的动作
const actions = {
    increment(context, value) {
        context.commit('INCREMENT', value)
    },
    decrement(context, value) {
        context.commit('DECREMENT', value)
    },
    incrementOdd(context, value) {
        if (context.state.sum % 2) {
            context.commit('INCREMENT', value)
        }
    },
    incrementWait(context, value) {
        setTimeout(() => {
            context.commit('INCREMENT', value)
        }, 1000)
    },
}
// mutations - 用于操作数据(state)
const mutations = {
    INCREMENT(state, value) {
        state.sum += value
    },
    DECREMENT(state, value) {
        state.sum -= value
    },
}
// state - 用于存储数据
const state = {
    sum: 0,
}

// 创建store 暴露store
export default new Vuex.Store({
    actions,
    mutations,
    state
});

Count.vue

<template>
  <div>
    <h1>当前求和为: {{ $store.state.sum }}</h1>
    <select v-model.number="number">
      <option :value="1">1</option>
      <option :value="2">2</option>
      <option :value="3">3</option>
    </select>
    <button @click="increment">+</button>
    <button @click="decrement">-</button>
    <button @click="incrementOdd">当前求和为奇数再加</button>
    <button @click="incrementWait">等一等再加</button>
  </div>
</template>
<script>
export default {
  name: `Count`,
  data() {
    return {
      number: 1
    }
  },
  methods: {
    increment() {
      this.$store.dispatch('increment', this.number)
    },
    decrement() {
      this.$store.dispatch('decrement', this.number)
    },
    incrementOdd() {
      this.$store.dispatch('incrementOdd', this.number)
    },
    incrementWait() {
      this.$store.dispatch('incrementWait', this.number)
    },
  },
  mounted() {
  }
}
</script>

<style>
button {
  margin-left: 10px;
}
</style>

5.2 没必要走actions

在这里插入图片描述

6 context 中的dispatch在actions中的链式调用

在这里插入图片描述
在这里插入图片描述
我们可以这样操作一个复杂的actions业务逻辑:
在这里插入图片描述

7 注意,业务逻辑写在哪里需要因地制宜

8 getters - 当state中的数据需要经过加工后再使用时,可以使用getters加工

在这里插入图片描述

9 一个问题:$store.state $store.getters重复书写的问题 - mapStatemapGetters

在这里插入图片描述
我们的解决方案是:借助mapState生成计算属性
在这里插入图片描述

    // 借助mapState生成计算属性,从state中读取数据(对象写法)
    ...mapState({
      sum: 'sum',
      school: 'school',
      subject: 'subject'
    })
    // 借助mapState生成计算属性,从state中读取数据(数组写法)
     ...mapState(['sum', 'school', 'subject'])

对于:$store.getters.bigSum

import {mapState, mapGetters} from 'vuex'

  computed: {
    ...mapGetters(['bigSum'])
  },

10 一个问题:$store.commit重复写的问题 - mapMutations

在这里插入图片描述

   <button @click="increment(number)">+</button>
   <button @click="decrement(number)">-</button>
import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'
  methods: {
    ...mapMutations({
      increment: 'INCREMENT',
      decrement: 'DECREMENT'
    }),
  },

11 一个问题:$store.dispatch重复写的问题 - mapActions

    <button @click="incrementOdd(number)">当前求和为奇数再加</button>
    <button @click="incrementWait(number)">等一等再加</button>
import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'

  methods: {
    ...mapActions(['incrementOdd', 'incrementWait'])
  },

12 多组件共享

在这里插入图片描述

13 namespace 的引入-这样就是一个封装的思路-模块化的思路

这里是使用 import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'
在这里插入图片描述
这里是使用传统的方法
在这里插入图片描述
我们调用后端服务获取数据:
在这里插入图片描述
在这里插入图片描述

14 最终源码

在这里插入图片描述

main.js

import Vue from 'vue'
import App from './App.vue'

Vue.config.productionTip = false

// 引入store
import store from "./store/index";

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

App.vue

<template>
  <div id="app">
    <Count/>
    <br/>
    <Person/>
  </div>
</template>
<script>

import Count from "@/components/Count";
import Person from "@/components/Person";

export default {
  name: `App`,
  data() {
    return {
    }
  },
  components: {Count, Person},
  methods: {},
}
</script>

<style>
</style>

index.js

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

// 引入Vue
import Vue from "vue";
// 引入vuex
import Vuex from 'vuex'
import axios from "axios";
import {nanoid} from "nanoid";
Vue.use(Vuex)

const addOptions = {
    namespaced: true,
    actions: {
        incrementOdd(context, value) {
            if (context.state.sum % 2) {
                context.commit('INCREMENT', value)
            }
        },
        incrementWait(context, value) {
            setTimeout(() => {
                context.commit('INCREMENT', value)
            }, 1000)
        },
    },
    mutations: {
        INCREMENT(state, value) {
            state.sum += value
        },
        DECREMENT(state, value) {
            state.sum -= value
        },
    },
    state: {
        sum: 0,
        school: '北京大学',
        subject: '软件工程',
    },
    getters: {
        bigSum(state) {
            return state.sum * 10
        }
    }
}
const personOptions = {
    namespaced: true,
    actions: {
        addPersonForWang(context, value) {
            if (value.name.indexOf('王') === 0) {
                context.commit('ADDPERSON', value)
            }
        },
        addPersonServer(context) {
            axios.get('https://api.uixsj.cn/hitokoto/get?type=social').then(
                response => {
                    let personNew = {
                        id: nanoid(),
                        name: response.data
                    }
                    context.commit('ADDPERSON', personNew)
                }, error => {
                    alert('请求失败')
                }
            )
        }
    },
    mutations: {
        ADDPERSON(state, value) {
            state.personList.unshift(value)
        }
    },
    state: {
        personList: [
            {id: '001', name: 'zhaoshuai-l1'},
            {id: '002', name: 'zhaoshuai-l2'},
            {id: '003', name: 'zhaoshuai-l3'},
        ]
    },
    getters: {
        getFirstNameFromList(state) {
            return state.personList[0].name
        }
    }
}
// 创建store 暴露store
export default new Vuex.Store({
    modules: {
        addOptions: addOptions,
        personOptions: personOptions
    }
});


Person.vue

<template>
  <div>
    <h2>列表中的第一个人的名字: {{ getFirstNameFromList }}</h2>
    <input v-model="name" type="text">
    <button @click="addPerson">添加</button>
    <button @click="addPersonForWang">添加一个姓王的人员</button>
    <button @click="addPersonServer">添加一个来自服务器的人员</button>
    <ul>
      <li v-for="item in personList" :value="item.name" :key="item.id">{{ item.name }}</li>
    </ul>
  </div>
</template>

<script>
import {nanoid} from 'nanoid'
import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'

export default {
  name: '',
  data() {
    return {
      name: ''
    }
  },
  methods: {
    addPerson() {
      let personNew = {
        id: nanoid(),
        name: this.name
      }
    this.$store.commit('personOptions/ADDPERSON', personNew)
    },
    addPersonForWang() {
      let personNew = {
        id: nanoid(),
        name: this.name
      }
      this.$store.dispatch('personOptions/addPersonForWang', personNew)
    },
    addPersonServer() {
      this.$store.dispatch('personOptions/addPersonServer')
    }

  },
  computed: {
    personList() {
      return this.$store.state.personOptions.personList
    },
    getFirstNameFromList() {
      return this.$store.getters['personOptions/getFirstNameFromList']
    }
  },
  mounted() {
    console.log(this)
  }
}
</script>

<style>
</style>

Count.vue

<template>
  <div>
    <h1>当前求和为: {{ sum }}</h1>
    <h1>当前求和 * 10 为: {{ bigSum }}</h1>
    <h1>School 为: {{ school }}</h1>
    <h1>Subject 为: {{ subject }}</h1>
    <select v-model.number="number">
      <option :value="1">1</option>
      <option :value="2">2</option>
      <option :value="3">3</option>
    </select>
    <button @click="increment(number)">+</button>
    <button @click="decrement(number)">-</button>
    <button @click="incrementOdd(number)">当前求和为奇数再加</button>
    <button @click="incrementWait(number)">等一等再加</button>
  </div>
</template>
<script>
import {mapState, mapGetters, mapMutations, mapActions} from 'vuex'

export default {
  name: `Count`,
  data() {
    return {
      number: 1
    }
  },
  methods: {
    ...mapMutations('addOptions', {
      increment: 'INCREMENT',
      decrement: 'DECREMENT'
    }),
    ...mapActions('addOptions', ['incrementOdd', 'incrementWait'])
  },
  computed: {
    ...mapState('addOptions', ['sum', 'school', 'subject']),
    ...mapState('personOptions', ['personList']),
    ...mapGetters('addOptions', ['bigSum'])
  }
}
</script>

<style>
button {
  margin-left: 10px;
}
</style>

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

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

相关文章

Kafka入门(七)

下面聊聊Kafka的配置参数&#xff0c;包括生产者的配置参数、Broker的配置参数、消费者的配置参数。 1、生产者配置参数 acks 该参数控制了生产者的消息发送确认机制&#xff0c;用于指定分区中必须有多少个副本成功接收到消息后生产者才会认为这条消息写入是成功的&#xff0c…

问题解决:利用GeoFabrik下载OSM数据

问题描述由于OpenStreetMap数据常常是.osmnx格式&#xff0c;需要在Linix系统上进行格式转换&#xff0c;比较不方便。GeoFabrik 下载平台提供了一个很好的直接下载.shp数据格式的途径&#xff0c;链接如下&#xff1a;【无需梯子】Geofabrik Download Server具体下载方法以Nep…

QT-项目创建

项目创建 注意: 项目创建&#xff0c;名称和路径不能包含中文路径。 创建窗口三大基类 QWidhetQMainWindowQDialog 1.1 项目文件介绍 mian.cpp 介绍 #include "mywidget.h" #include <QApplication> // QApplication 应用程序类// 程序入口 命令行变量数量 …

Zabbix6.2利用模板和自定义监控项监控华为AR3260路由器

1&#xff1a;登录路由器的WEB管理控制台。在系统管理中找到SNMP然后开启SNMP代理&#xff0c;SNMP的版本可以只选择v2c都选择也无所谓&#xff0c;然后点击新建一个团体。 2&#xff1a;团体名称输入默认的public即可&#xff0c;在WEB端显示的是乱码&#xff0c;但是不影响使…

SQL执行过程详解

1 、用户在客户端执行 SQL 语句时&#xff0c;客户端把这条 SQL 语句发送给服务端&#xff0c;服务端的进程&#xff0c;会处理这条客户端的SQL语句。 2 、服务端进程收集到SQL信息后&#xff0c;会在进程全局区PGA 中分配所需内存&#xff0c;存储相关的登录信息等。 3 、客…

国外SEO策略指南:确保你的网站排名第一!

如果你想在谷歌等搜索引擎中获得更高的排名并吸引更多的流量和潜在客户&#xff0c;那么你需要了解一些国外SEO策略。 下面是一些可以帮助你提高谷歌排名的关键策略。 网站结构和内容优化 谷歌的搜索算法会考虑网站的结构和内容。 因此&#xff0c;你需要优化网站结构&…

2379. 得到 K 个黑块的最少涂色次数

2379. 得到 K 个黑块的最少涂色次数 难度简单 给你一个长度为 n 下标从 0 开始的字符串 blocks &#xff0c;blocks[i] 要么是 W 要么是 B &#xff0c;表示第 i 块的颜色。字符 W 和 B 分别表示白色和黑色。 给你一个整数 k &#xff0c;表示想要 连续 黑色块的数目。 每一…

Python数据结构与算法篇(二)-- 数组常见题型与解题技巧

数组和链表代表着计算机最基本的两种存储形式&#xff1a;顺序存储和链式存储&#xff0c;所以他俩可以算是最基本的数据结构。数组是一种基础数据结构&#xff0c;可以用来处理常见的排序和二分搜索问题&#xff0c;典型的处理技巧包括对双指针、滑动窗口等&#xff0c;数组是…

想要将多个视频拼接在一起?如何把三个视频合成一个视频

从事短视频创作行业以来&#xff0c;总是存在着各种挑战。最开始&#xff0c;因为主要负责视频素材的搜集&#xff0c;所以每天虽忙但充实&#xff0c;最近逐步开始学习视频的剪辑工作&#xff0c;可把我难到了&#xff01;想要将多个视频拼接在一起&#xff1f;如何把三个视频…

「 Java 8 新特性 」Stream 中的 map、peek、foreach 方法的区别

「 Java 8 新特性 」Stream 中的 map、peek、foreach 方法的区别 文章参考&#xff1a; 面试官问&#xff1a;Stream 中的 map、peek、foreach 方法的区别&#xff1f;傻傻分不清楚。。 stream中的map,peek,foreach的区别 一、概述 在学习java 8的stream api时&#xff0c;我们…

Java【数据结构入门OJ题33道】——力扣刷题记录1

文章目录第一天存在重复元素最大子数组和第二天两数之和合并两个有序数组第三天两个数组的交集买卖股票最佳时机第四天重塑矩阵杨辉三角第五天有效的数独矩阵置零第六天字符串中第一个唯一字符救赎金第七天判断链表是否有环合并两个有序链表移除链表元素第八天反转链表删除重复…

c++面试技巧-高级应用篇

1.面试官&#xff1a;什么是文件流&#xff1f; 应聘者&#xff1a;写入文件或者从文件读出的数据流称之为文件流。 2.面试官&#xff1a;文件流的类时如何划分的&#xff1f; 应聘者&#xff1a;当C对文件进行处理时&#xff0c;需要包括头文件iostream和fstream。fstream头…

游戏逆向之游戏技能分析

角色的当前技能列表往往都是从系统的技能库中进行筛选而组成的&#xff0c;而这个筛选的过程大多非常的复杂&#xff0c;经过的代码和临时结构体的传递也非常的多&#xff0c;所以在分析技能对象来源的时候常常要将OD和CE配合来使用。下面我们来分析下《天堂2》的技能列表。 首…

拼多多存在多种重大风险

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 中国电子商务市场发展迅速 拼多多&#xff08;PDD&#xff09;目前已经成为了在中国发展最快的电子商务平台之一。随着拼多多、阿里巴巴(BABA)和京东(JD)等市场主要参与者的竞争&#xff0c;中国电子商务正在迅速发展。 由…

安装SceneBuilder时出现Verify that you have access to that directory报错,解决方法之一

1、问题描述 安装SceneBuilder时出现Error writing to file...Verify that you have access to that directory报错&#xff0c;如图 2、解决步骤 1&#xff09;SceneBuilder-19.0.0.msi可从参考教程链接1获取&#xff0c;下载到文件夹D:\LcSceneBuilder\SceneBuilder-19.0.0…

Spring Cloud融合gateway自带GatewayFilter使用 | Spring Cloud 15

一、Spring Cloud Gateway内置GatewayFilter 路由过滤器允许以某种方式修改传入的 HTTP 请求或传出的 HTTP 响应。路由过滤器的范围是特定路由。Spring Cloud Gateway 包括许多内置的 GatewayFilter 工厂。 官网地址&#xff1a;https://docs.spring.io/spring-cloud-gateway…

MATLAB与图像处理的那点小事儿~

目录 一、学习内容 二、matlab基本知识 三、线性点运算 四、非线性点运算&#xff0c;伽马矫正 五、直方图 1、直方图均衡化 &#xff08;1&#xff09;使用histep函数实现图像均衡化 &#xff08;2&#xff09;使用自行编写的均衡化函数实现图像均衡化 2、直方图规定…

Ansys Zemax | 如何使用 OpticStudio 非序列优化向导

本文描述了如何使用 OpticStudio 非序列优化向导创建常见的评价函数类型&#xff0c;以及创建用于匹配导入图像文件的目标能量分布评价函数。&#xff08;联系我们获取文章附件&#xff09; 简介 在非序列模式下优化光学系统通常比在序列模式下的优化更复杂、更耗时。下期我们…

TencentOS tiny 移植到STM32F103教程

一、移植前的准备工作1. STM32的裸机工程模板这个可以自己创建&#xff08;创建过程参考之前的STM32裸机工程也有工程网盘在上面&#xff09;2.下载TencentOS tiny 源码TencentOS tiny的源码可从TencentOS tiny GitHub仓库地址https://github.com/Tencent/TencentOS-tiny下载&a…

无人机动力测试台:150kg级-Flight Stand 150

“飞行汽车”被人们广泛视为下一代交通工具。从2022年初至今&#xff0c;eVTOL&#xff08;电动垂直起降飞行器&#xff09;产业乃至整个UAM&#xff08;城市间空中交通&#xff09;市场呈现爆发式增长——各国航空制造企业纷纷入局&#xff0c;不断有新产品问世、试飞或试运行…