vue中全局状态存储 pinia和vuex对比 pinia比vuex更香 Pinia数据持久化及数据加密

news2024/11/25 14:57:59

                                              在这里插入图片描述

前言

毕竟尤大佬都推荐使用pinia,支持vue2和vue3!

如果熟悉vuex,花个把小时把pinia看一下,就不想用vuex了

  • 支持选项式api和组合式api写法
  • pinia没有mutations,只有:state、getters、actions
  • pinia分模块不需要modules
  • pinia 没有命名空间模块。
  • pinia 无需动态添加(底层通过 getCurrentInstance 获取当前 vue 实例进行关联)。
  • pinia 是平面结构(利于解构),没有嵌套,可以任意交叉组合。
    在这里插入图片描述

先介绍两个官网:
pinia官网

保持pinia的数据持久化用pinia-plugin-persistedstate (推荐使用这个,其他的也可以,如:pinia-plugin-persist可能会遇到bug)
pinia-plugin-persistedstate 官网

如果需要对本地存储加密的话使用 secure-ls 插件:原理是重构localStorage 和 sessionStorage 的setItem和getItem方法,同理vuex里面也可以使用
secure-ls 网址

安装

npm install pinia pinia-plugin-persistedstate

创建store

在根目录下创建store文件夹,新建index.ts文件


import { createPinia } from 'pinia';
import createPersistedState from 'pinia-plugin-persistedstate'; //引入pinia数据持久化插件
const pinia = createPinia();
pinia.use(createPersistedState);
export default pinia;


挂载

在这里插入图片描述

Pinia数据持久化及数据加密

这里使用的pinia数据的持久化,和本地加密存储
pinia中不在用modules分模块,简单理解一个ts文件一个模块吧,注意每个模块尽量统一风格,导出变量use开头
下面是个示例:
store文件夹中的 counter.ts 这里用到了,数据持久化和数据加密

import { defineStore } from 'pinia'; // 定义一个pinia的一个模块
//这里是练习pinia用的
import type { StorageLike } from 'pinia-plugin-persistedstate';
import SecureLS from 'secure-ls';
// encryptionSecret:自定义密钥
const ls = new SecureLS({
  isCompression: false,
  encryptionSecret: '38c31684-d00d-30dc-82e0-fad9eec46d1d',
});
const st: StorageLike = {
  setItem(key: string, value: string) {
    ls.set(key, value);
  },
  getItem(key: string): string | null {
    return ls.get(key);
  },
};

//'第一个参数是id标识,pinia的名字','第二个参数是个对象'
export const useCounterStore = defineStore('counter', {
  state: () => {
    return {
      count: 10,
      msg: '菠萝头的使用',
      price: 100,
    };
  },
  actions: {
    add() {
      this.count += 1;
    },
    del() {
      this.count -= 1;
    },
    promiseIncrement() {
      return new Promise((resolve) => {
        setTimeout(() => {
          this.count++;
          resolve(this.count);
        }, 1000);
      });
    },
    priceAdd(a: any, b: any) {
      console.log(a, b);

      this.price++;
    },
  },
  getters: {
    getCount(state) {
      //在使用时,把方法名当属性用就行
      return state.count * state.price;
    },
  },
  // persist: true, 默认存储在 localStorage上面的
  persist: {
    storage: st, // 上面申明的类型
    // storage: localStorage,
    // key: 'counter',
  },
});

moduleIdentifiers.ts文件,无加密

import { defineStore } from 'pinia';

let moduleIdentifiers: Array<String> = [];

export const useModuleIdentifiers = defineStore('moduleIdentifiers', {
  state: () => {
    return {
      moduleIdentifiers,
    };
  },
  actions: {
    setModuleIdentifiers(aside: Array<String>) {
      this.moduleIdentifiers = aside;
    },
  },
  getters: {},
  persist: true,
  // persist: {
  //   storage: localStorage,
  // },
});

页面中使用

<template>
  <a-layout class="layout">
    <Header />
    <a-layout-content class="main-container">
      <a-button @click="back">返回上一页</a-button>

      <div class="title">
        <h1>pinia</h1>
        <div class="button-box">
          <a-button type="primary" @click="test">去练习vuex</a-button>
          <a-button type="primary" @click="changePina(1)">add同步</a-button>
          <a-button type="primary" @click="changePina(2)">promise异步</a-button>
          <a-button type="primary" @click="changePina(3)">test</a-button>
          <a-button type="primary" @click="changePina(5)">$patch</a-button>
          <a-button type="primary" @click="changePina(6)">getters</a-button>
          <a-button type="primary" @click="changePina(4)">reset</a-button>
        </div>
      </div>
      <div class="card">
        <a-button>stata.count: {{ count }}</a-button>
        <a-button>stata.msg: {{ msg }}</a-button>
        <a-button>stata.price: {{ price }}</a-button>
      </div>
    </a-layout-content>
  </a-layout>
</template>

<script lang="ts" setup>
import { reactive, computed } from 'vue';
import { useRouter } from 'vue-router';

import Header from '/@/components/Header/index.vue';

import { useCounterStore } from '/@/store/counter';

import { storeToRefs } from 'pinia';
let pinia = useCounterStore();

//使用storeToRefs解构保持响应性,当然也可以直接结构,根据具体情况而定
let { count, price, msg, getCount } = storeToRefs(pinia);
// const count = computed(() => pinia.count)
const changePina = (type: number) => {
  if (type == 1) {
    pinia.add();
  } else if (type == 2) {
    pinia.promiseIncrement();
  } else if (type == 3) {
    pinia.priceAdd(1, 2);
  } else if (type == 4) {
    pinia.$reset();
  } else if (type == 5) {
    pinia.$patch((state) => {
      state.count += 2;
      state.price += 2;
      state.msg = '$patch批量修改';
    });
  } else if (type == 6) {
    console.log(pinia.getCount);
    //如果想使用解构,一定要用 storeRoRefs
    console.log(getCount);
  }
};
const router = useRouter();

const test = () => {
  router.push('/ncov/test');
};
const back = () => {
  router.push('/ncov/task');
};
</script>

<style lang="less" scoped>
.layout {
  min-width: 1400px;
  min-height: 100vh;
}

.main-container {
  flex-direction: row;
  min-height: calc(100vh - 76px);
  padding: 19px 70px;
  background-color: @bg-color;
}

.title {
  font-weight: 600;
  font-size: 24px;
  margin: 30px 0;
  .flex-type(space-between);

  :deep(.ant-btn) {
    margin-left: 10px;
  }
}

.card {
  background-color: @white-color;
  padding: 30px;

  :deep(.ant-btn) {
    margin-left: 10px;
  }
}

.select-box {
  margin-bottom: 40px;
  .flex-type(flex-start);

  :deep(.ant-select) {
    width: 120px;
    margin-right: 40px;
  }
}

:deep(.ant-cascader-menu-item:hover) {
  background: @bg-color;
}

:deep(.ant-input:focus, .ant-input:hover, .ant-cascader-input:hover) {
  border-color: rgba(122, 132, 198, 1);
}
</style>

详情解析

上面的代码已经包含了 State、Getters、Actions(同步/异步)、$reset$patch
State:存数据
Getters:计算属性
Actions:更改state的值
$reset:重置state到最初状态
$patch:批量修改state的值
其他还有替换所有state的值等,参考官网

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

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

相关文章

20个程序员接单平台分享

这题我会&#xff01;接单软件那么多&#xff0c;找到适合自己的最重要&#xff01; V2EX https://www.v2ex.com/ 先给一个“非正常选项”&#xff0c;v2ex上有一个“酷工作”板块&#xff0c;运气好的话可以在这里找到不错的单子&#xff0c;最重要的是带你开启新世界的大门…

新型声学攻击通过键盘击键窃取数据,准确率高达 95%

来自英国大学的一组研究人员训练了一种深度学习模型&#xff0c;该模型可利用麦克风记录并分析键盘击键的声音&#xff0c;以此来窃取目标设备中的数据&#xff0c;准确率高达 95%。 不同于其他需要特殊条件并受到数据速率和距离限制的旁道攻击&#xff0c;由于现有大量场景都拥…

LeetCode:Hot100的python版本

94. 二叉树的中序遍历

Python爬虫的Selenium(学习于b站尚硅谷)

目录 一、Selenium  1.为什么要学习Selenium  &#xff08;1&#xff09;什么是Selenium  &#xff08;2&#xff09;为什么使用selenium?  &#xff08;3&#xff09;代码演示 2. selenium的基本使用  &#xff08;1&#xff09;如何安装selenium  &#xff08;2…

jmeter如何压测和存储

一、存储过程准备&#xff1a; 1、建立一个空表&#xff1a; 1 CREATE TABLE test_data ( id NUMBER, name VARCHAR2(50), age NUMBER ); 2、建立一个存储过程&#xff1a; 1 2 3 4 5 6 7 8 9 CREATE OR REPLACE PROCEDURE insert_test_data (n IN NUMBER) AS BEGIN --E…

【工程优化问题】基于多种智能优化算法的压力容器设计问题研究(Matlab代码实现)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

Linux内核应该怎么去学习?

通过阅读源码来学习操作系统要注意区分共性与平台特性。 1. 中断响应是共性&#xff0c;8259 中断控制器和 IDT 是 x86 的特性。 2. 虚拟内存管理是共性&#xff0c;x86 的 GDT 和 LDT 是特性&#xff0c;而且现在的系统也只是走个过场而已。 3. 任务调度与上下文切换是共性&am…

44.实现爱尔兰B公式计算并输出表格(matlab程序)

1.简述 1.话务量定义 话务量指在一特定时间内呼叫次数与每次呼叫平均占用时间的乘积。 话务量反映了电话负荷的大小&#xff0c;与呼叫强度和呼叫保持时间有关。呼叫强度是单位时间内发生的呼叫次数&#xff0c;呼叫保持时间也就是占用时间。 话务量计算方法 话务量公式为…

智安网络|恶意软件在网络安全中的危害与应对策略

恶意软件是指一类具有恶意目的的软件程序&#xff0c;恶意软件是网络安全领域中的一个严重威胁&#xff0c;给个人用户、企业和整个网络生态带来巨大的危害。通过潜伏于合法软件、邮件附件、下载链接等途径传播&#xff0c;破坏用户计算机系统、窃取敏感信息、进行勒索等不法行…

C语言经典小游戏之扫雷(超详解释+源码)

“少年气&#xff0c;是历尽千帆举重若轻的沉淀&#xff0c;也是乐观淡然笑对生活的豁达&#xff01;” 今天我们学习一下扫雷游戏怎么用C语言来实现&#xff01; 扫雷小游戏 1.游戏介绍2.游戏准备3.游戏实现3.1生成菜单3.2游戏的具体实现3.2.1初始化棋盘3.2打印棋盘3.3布置雷…

Linux root用户执行修改密码命令,提示 Permission denied

问题 linux系统中&#xff08;ubuntu20&#xff09;&#xff0c;root用户下执行passwd命令&#xff0c;提示 passwd: Permission denied &#xff0c;如下图&#xff1a; 排查 1.执行 ll /usr/bin/passwd &#xff0c;查看文件权限是否正确&#xff0c;正常情况是 -rwsr-xr…

VAE、 EM、KL散度

文章目录 VAEVAE额外的损失函数 EMKL散度 VAE 左图相当于变量x&#xff0c;右图相当于z 假如在AE中&#xff0c;一张满月的图片作为输入&#xff0c;模型得到的输出是一张满月的图片&#xff1b;一张弦月的图片作为输入&#xff0c;模型得到的是一张弦月的图片。当从满月的code…

SpringBoot复习:(22)ConfigurationProperties和@PropertySource配合使用及JSR303校验

一、配置类 package cn.edu.tju.config;import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.context.annotation.PropertySource; import org.springframework.stereotype.Component;Component ConfigurationPropertie…

C++初阶——函数重载

前言&#xff1a;C中除了可以在不同的命名空间中使用同名函数&#xff0c;还有一种支持在同一个作用域中同名函数的方式——函数重载。 函数重载 一.什么是函数重载&#xff1f;二.函数重载的3种规则三.特殊情况 一.什么是函数重载&#xff1f; C允许同样同一作用域中声明几个功…

IPWorks OFX Delphi Edition Crack

IPWorks OFX Delphi Edition Crack IPWorks OFX由可以访问电子交易信息的组件组成&#xff0c;并包括银行转账和付款提交等功能。这些组件使应用程序开发人员能够构建包含更快、更准确的交易对账、发送即时交易通知以及完全关闭支付和会计之间循环的解决方案。 IPWorks OFX功能…

请求接口时报异常:org.springframework.web.multipart.MultipartException

请求接口时报异常&#xff1a;org.springframework.web.multipart.MultipartException: Current request is not a multipart request 检查后发现自己忘记传文件参数 添加参数后请求正常。

软件测试的生命周期

目录 软件测试&软件开发生命周期 如何描述一个bug? 如何定义bug的级别 bug的生命周期 测试的执行和BUG管理 产生争执怎么办&#xff08;处理人际关系&#xff09; 软件测试&软件开发生命周期 需求阶段 --测试人员需要了解需求, 对需求进行分解得出测试需求 计…

python的virtualenv虚拟环境无法激活activate

目录 问题描述&#xff1a; 解决办法&#xff1a; 解决结果&#xff1a; 问题描述&#xff1a; PS D:\pythonProject\pythonProject\DisplayToolLibs\venv\Scripts> .\activate .\activate : 无法加载文件 D:\pythonProject\pythonProject\DisplayToolLibs\venv\Scripts\…

全网最牛,在Linux系统上安装Git详细步骤,看这一篇就够了...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 yum安装 1、在Li…

学习笔记-JVM-对象结构及生命周期

申明&#xff1a;文章内容是本人学习极客时间课程所写&#xff0c;文字和图片基本来源于课程资料&#xff0c;在某些地方会插入一点自己的理解&#xff0c;未用于商业用途&#xff0c;侵删。 原资料地址&#xff1a;课程资料 对象的创建流程 常量池检查:检查new指令是否能在常…