稀土掘金技术社区 02月12日
大菠萝(Pinia)符合真香定律
index_new5.html
../../../zaker_core/zaker_tpl_static/wap/tpl_guoji1.html

 

本文介绍了Pinia,一个为Vue设计的轻量级状态管理库,它既能在Vue3中使用,也兼容Vue2。Pinia通过Store来管理全局状态,Store包含state、getters和actions三个核心概念,分别对应数据、计算属性和方法。文章详细讲解了如何安装、创建和使用Pinia,并对比了Option对象形式和Setup函数形式的Store配置。此外,还介绍了Pinia提供的如$reset、$patch、$state、$subscribe等常用操作,以及actions的异步操作、组合操作和数据持久化功能,展示了Pinia在现代Vue开发中的灵活性和便捷性。

🍍Pinia是一个Vue的状态管理库,允许跨组件/页面共享状态,它不仅能在Vue3中使用,还兼容Vue2,为开发者提供了便利。

🛠️Pinia的核心概念是Store,通过`defineStore`创建,可以采用Option对象或Setup函数两种形式配置state(状态)、getters(计算属性)和actions(方法),Option形式更直观,Setup形式更灵活,还可以创建监听器。

🔄Pinia提供多种状态操作方法,如`$reset`重置状态,`$patch`批量更新状态,`$state`替换整个状态,以及`$subscribe`监听状态变化,方便开发者更好地管理应用状态。

💾Pinia的action支持异步操作、组合操作,并且可以通过安装`pinia-plugin-persistedstate`插件实现数据持久化,即使刷新页面数据也不会丢失。

原创 ys指风不买醉 2025-02-09 09:00 重庆

点击关注公众号,“技术干货” 及时达!

前言

最近在开发Vue3项目时,遇到了多层跨域数据传输的问题。起初,我用propsemit实现了爷孙组件之间的数据传递,效果不错。子组件通过defineEmit定义方法,父组件通过emit调用这些方法,而defineProps则负责接收参数,一切井然有序。

但问题出现在兄弟组件之间的数据传递上。兄弟组件的数据由父组件管理,两者不会直接修改数据,每次使用都从父组件独立获取属性或方法。然而,这些兄弟组件并非“堂兄弟”,而是“表兄弟”,数据传输变得复杂。

有人可能会想到用injectprovide实现跨层级传递,但随着项目规模扩大,数据来源不明确,维护起来会很麻烦。

这时,Vuex似乎是个解决方案,但今天的主角是更轻量级的Pinia,也就是“大菠萝”?

大菠萝Pinia是啥?

Pinia 是 Vue 的存储库,它允许您跨组件/页面共享状态。

这是Pinia官网对它的定义。Vuex用起来还是有点麻烦,React里的Context倒是还不错。而Pinia不仅能在Vue3中使用,还兼容Vue2,简直是开发者的福音。下面是 Pinia官网对自身的评价,看起来它对自己的定位非常清晰。

Pinia 常规操作

通过常用的包管理器进行安装:

yarn add pinia// 或者npm install pinia

安装完成后,我们需要将Pinia挂载到Vue应用中。也就是说,我们需要创建一个根存储并传递给应用。我们需要修改main.js,引入Pinia提供的createPinia方法:

import { createApp } from 'vue'import App from './App.vue'import { createPinia } from 'pinia'
const pinia = createPinia()const app = createApp(App)app.use(pinia)app.mount('#app')

创建Pinia实例,并在根组件中use一下,这样整个应用就可以使用Pinia来管理状态了。

Store

Store是Pinia的核心概念,它是一个中央数据仓库,专门用于状态管理。这里存放的数据是整个应用可以访问的全局数据,而组件内部的数据不应该放入Store。

使用Pinia提供的defineStore方法创建一个全局仓库。defineStore接收两个参数:

defineStore配置store属性时有两种写法,一种是Option对象形式,一种是Setup函数形式。

对象形式(Option store)

Store有三个核心概念:「state」「getters」「actions」,我们可以将它们类比为组件中的“「数据」”、“「计算属性」”和“「方法」”。(直观,推荐使用)

// 中央管理store import { defineStore } from 'pinia'import { ref } from 'vue'
// 第一个参数:唯一ID;第二个:一个对象,其他配置项export const useCounterStore = defineStore('counter', { // 推荐使用完整类型推断的箭头函数 state: () => { return { count: ref(0), isLogin: ref(false), } }, actions: { add() { this.count++ } }})

函数形式(setup store)

与 Vue 组合式 API 的 setup 函数 相似,我们可以传入一个函数,该函数定义了一些响应式属性和方法,并且返回一个带有我们想暴露出去的属性和方法的对象。

在 Setup Store 中:

可见,函数式相比对象式更灵活,还可以创建监听器

import { defineStore } from 'pinia';import { ref } from 'vue';
export const useCounterStore = defineStore('counter', () => { // 定义状态 const count = ref(0); const isLogin = ref(false);
// 定义 actions function add() { count.value++; }
// 返回状态和 actions return { count, isLogin, add, };});

使用Store

下面都说Option对象形式创建的store

可以在任意组件中引入定义的Store来进行使用。比如在App.js中:

<template>  <div>    <p>数量: {{ counterStore.count }}</p>    <p>是否登入: {{ counterStore.isLogin }}</p>    <button @click="counterStore.add">增加Add</button>  </div></template>
<script setup>import { useCounterStore } from './stores/counter';
const counterStore = useCounterStore();</script>

实例化后,就可以使用Store里面的stategettersactions中定义的任何属性了。

可能有人会说每次都用实例counterStore.去点方法/属性,有点麻烦。这时候大菠萝pinia又给舔了一嘴——使用「解构」直接

解构与直接访问

一般来说,我们实例化Store后,直接进行读取和写入操作即可。

// CompA.js<template>  <div>    {{ count }}    <button @click="add">B点我{{ count }}</button>    <br>    <button @click="counterStore.add">A点我{{ counterStore.count }}</button>  </div></template>
<script>import { useCounterStore } from '../store/counter'const counterStore = useCounterStore()const { count, add } = useCounterStore();</script>

这里有个问题,如果直接解构,会发现数据失去了响应式。

这时候,我们可以使用toRef一个一个解构,或者使用toRefs一次性解构所有属性或方法。Pinia还有一个自带的解构状态storeToRefs,但是不包括方法。如果你使用的是 Vue 3 的 <script setup> 语法,可以直接解构和使用 Pinia store

// 使用 toRefs 一次性解构所有属性const { count, add } = toRefs(counterStore);

// 使用 toRef 逐个解构const count = toRef(counterStore, 'count');const add = toRef(counterStore, 'add');
// 使用 storeToRefs 解构状态const { count } = storeToRefs(counterStore);
// setup语法糖下,直接解构方法const { add } = counterStore;

State

其他常用的操作还有:$reset$patch$state$subscribe等,这些方法可以帮助我们更好地管理状态。

import { useCounterStore } from './store/counter';const counterStore = useCounterStore();
counterStore.count = 10;// reset重置状态为初始值counterStore.$reset(); // count 恢复为初始值 0

$patch 可以一次性更新多个状态,支持对象或函数形式。

// 对象形式counterStore.$patch({  count: 5,  isLogin: true,});
// 函数形式counterStore.$patch((state) => { state.count += 1; state.isLogin = false;});

$state 用于替换整个 state 对象。

counterStore.$state = {  count: 100,  isLogin: true,};

$subscribe 可以监听 state 的变化,适合做一些副作用操作。

counterStore.$subscribe((mutation, state) => {  console.log('状态变化了:', state.count);});
// 修改状态counterStore.count = 20; // 控制台输出: "状态变化了: 20"

Actions

直接通过 this 访问 stategetters 和其他 action,不需要像vuex每次都去使用 context。同时可以直接修改状态,不需要像vuex通过 mutation

Pinia 的 action 天然支持异步操作,比如 async/await 或 Promise,无需额外配置。

actions: {  async fetchData() {    const data = await api.getData();    this.data = data; // 直接修改状态  }}

除了异步,直接修改状态,调用其他action和访问getters这4项操作外,Pinia 的 action 还有以下能力:

actions: {  async login(credentials) {    const user = await api.login(credentials); // 异步操作    this.user = user; // 直接修改状态    this.setLoginStatus(true); // 调用其他 action  },  setLoginStatus(status) {    this.isLoggedIn = status; // 直接修改状态  }}

「数据持久化」

store中的数据,刷新页面后就丢失了,如果想保留这些数据,就要用到数据持久化了。

推荐使用 「pinia-plugin-persistedstate」

yarn add pinia-plugin-persistedstate
import { createPinia } from 'pinia'import piniaPluginPersistedstate from 'pinia-plugin-persistedstate'
const pinia = createPinia()pinia.use(piniaPluginPersistedstate)

小结

Pinia 的设计让状态管理更加直观和灵活,特别适合现代 Vue 开发!还是不得不说:Vuex太重,Pinia(大菠萝)轻巧好用,真香!

点击关注公众号,“技术干货” 及时达!

阅读原文

跳转微信打开

Fish AI Reader

Fish AI Reader

AI辅助创作,多种专业模板,深度分析,高质量内容生成。从观点提取到深度思考,FishAI为您提供全方位的创作支持。新版本引入自定义参数,让您的创作更加个性化和精准。

FishAI

FishAI

鱼阅,AI 时代的下一个智能信息助手,助你摆脱信息焦虑

联系邮箱 441953276@qq.com

相关标签

Pinia Vue3 状态管理 轻量级
相关文章