Vuex 的核心概念:State, Mutations, Actions, Getters

Vuex 的核心概念:State, Mutations, Actions, Getters

Vuex 是 Vue.js 的官方状态管理库,提供了集中式的状态管理机制。它的核心概念包括 State(状态)、Mutations(变更)、Actions(动作)和 Getters(获取器)。以下是对这些概念的详细介绍及相应的代码示例。

State(状态)

State 是存储应用共享状态的地方。在 Vuex 中,State 是一个对象,包含应用中需要共享的所有数据。组件可以从 store 中读取状态,当状态发生变化时,依赖这些状态的组件会自动更新。

示例:

javascript 复制代码
// store.js
import { createStore } from 'vuex';

const store = createStore({
  state() {
    return {
      count: 0,
      todos: [
        { id: 1, text: 'Learn Vuex', done: true },
        { id: 2, text: 'Build a project', done: false },
      ],
    };
  },
});

export default store;

在组件中访问 state:

javascript 复制代码
// MyComponent.vue
<template>
  <div>
    <p>Count: {{ count }}</p>
    <ul>
      <li v-for="todo in todos" :key="todo.id">
        {{ todo.text }} - {{ todo.done ? 'Done' : 'Pending' }}
      </li>
    </ul>
  </div>
</template>

<script>
import { mapState } from 'vuex';

export default {
  computed: {
    ...mapState(['count', 'todos']),
  },
};
</script>

Mutations(变更)

Mutations 是更改 Vuex 中 state 的唯一方法。每个 mutation 都有一个字符串的事件类型(type)和一个回调函数(handler)。回调函数接收 state 作为第一个参数,可以直接修改 state。需要注意的是,mutations 必须是同步函数。

示例:

javascript 复制代码
// store.js
const store = createStore({
  state() {
    return {
      count: 0,
    };
  },
  mutations: {
    increment(state) {
      state.count++;
    },
    decrement(state) {
      state.count--;
    },
  },
});

在组件中提交 mutation:

javascript 复制代码
// MyComponent.vue
<template>
  <div>
    <p>Count: {{ count }}</p>
    <button @click="increment">Increment</button>
    <button @click="decrement">Decrement</button>
  </div>
</template>

<script>
import { mapState } from 'vuex';

export default {
  computed: {
    ...mapState(['count']),
  },
  methods: {
    increment() {
      this.$store.commit('increment');
    },
    decrement() {
      this.$store.commit('decrement');
    },
  },
};
</script>

Actions(动作)

Actions 类似于 mutations,不同之处在于:

  • Action 提交的是 mutation,而不是直接变更状态。
  • Action 可以包含任意异步操作。

示例:

javascript 复制代码
// store.js
const store = createStore({
  state() {
    return {
      count: 0,
    };
  },
  mutations: {
    increment(state) {
      state.count++;
    },
  },
  actions: {
    incrementAsync({ commit }) {
      setTimeout(() => {
        commit('increment');
      }, 1000);
    },
  },
});

在组件中分发 action:

javascript 复制代码
// MyComponent.vue
<template>
  <div>
    <p>Count: {{ count }}</p>
    <button @click="incrementAsync">Increment Async</button>
  </div>
</template>

<script>
import { mapState } from 'vuex';

export default {
  computed: {
    ...mapState(['count']),
  },
  methods: {
    incrementAsync() {
      this.$store.dispatch('incrementAsync');
    },
  },
};
</script>

在上述示例中,incrementAsync action 等待 1 秒后提交 increment mutation,从而异步地增加计数器的值。

Getters(获取器)

Getters 类似于组件的计算属性,用于从 state 中派生出一些状态。它们可以对 state 进行处理并返回结果,且结果会被缓存,只有当相关的 state 发生变化时才会重新计算。

示例:

javascript 复制代码
// store.js
const store = createStore({
  state() {
    return {
      todos: [
        { id: 1, text: 'Learn Vuex', done: true },
        { id: 2, text: 'Build a project', done: false },
      ],
    };
  },
  getters: {
    doneTodos: (state) => {
      return state.todos.filter((todo) => todo.done);
    },
    doneTodosCount: (state, getters) => {
      return getters.doneTodos.length;
    },
  },
});

在组件中访问 getters:

javascript 复制代码
// MyComponent.vue
<template>
  <div>
    <p>Done Todos Count: {{ doneTodosCount }}</p>
    <ul>
      <li v-for="todo in doneTodos" :key="todo.id">
        {{ todo.text }}
      </li>
    </ul>
  </div>
</template>

<script>
import { mapGetters } from 'vuex';

export default {
  computed: {
    ...mapGetters(['doneTodos', 'doneTodosCount']),
  },
};
</script>

在上述示例中,doneTodos getter 返回已完成的待办事项列表,doneTodosCount getter 返回已完成的待办事项数量。

通过合理地使用 Vuex 的这些核心概念,可以使 Vue.js 应用的状态管理更加清晰、可维护和可预测。

相关推荐
工业互联网专业16 分钟前
基于springboot+vue的数码产品抢购系统
java·vue.js·spring boot·毕业设计·源码·课程设计·数码产品抢购系统
Data_Adventure21 分钟前
使用CLINE快速生成一个3D展厅
vue.js·three.js·cline
周星星日记1 小时前
10.vue3中组件实现原理(上)
前端·vue.js·面试
Mikey_n1 小时前
Vue + Spring Boot 整合全解析
前端·vue.js·spring boot
xiaoyan20152 小时前
electron35+deepseek-v3桌面端ai聊天模板
vue.js·electron·deepseek
Lin_熊米2 小时前
仿 ElementUI 搭建自己的 vue 组件库
前端·vue.js·elementui
Bl_a_ck2 小时前
npm、nvm、nrm
前端·vue.js·npm·node.js·vue
diang2 小时前
Vue3 + Ant Design 实现 Excel 模板导入表格数据
前端·vue.js
doria小鱼2 小时前
Vue3+Swiper实现PC端横向滑动拖拽
前端·vue.js
李梦晓2 小时前
@vueuse/motion、motion-v、@motionone/vue三个动画库的区别和联系
前端·vue.js