1、创建项目(vite+vue+ts)
复制
2、安装pinia
复制
3、使用
1、引入注册main.ts
| import { createApp } from 'vue' |
| import App from './App.vue' |
| import { createPinia } from 'pinia' |
| const pinia = createPinia() |
| const app=createApp(App) |
| app.use(pinia).mount('#app') |
复制
2、初始化仓库Store(存储是使用定义的defineStore()
,并且它需要一个唯一的名称,作为第一个参数传递)
新建store/main.ts
| import { defineStore } from 'pinia' |
| |
| export const mainStore = defineStore('main',{ |
| state:()=>{ |
| return { |
| count:100, |
| price:250 |
| } |
| }, |
| getters:{ |
| doubleCount():number{ |
| return this.count*2 |
| }, |
| doublePrice():number{ |
| return this.price*2 |
| } |
| }, |
| actions:{ |
| changeStoreData(){ |
| this.count+=1 |
| this.price+=1 |
| }, |
| } |
| }) |
复制
3、页面使用

| <script setup lang="ts"> |
| import { ref } from 'vue' |
| import { storeToRefs } from 'pinia'; |
| import { mainStore } from '../store/main'; |
| const store = mainStore() |
| let { count,doubleCount,price,doublePrice } = storeToRefs(store) |
| defineProps<{ msg: string }>() |
| const changeData = ()=>{ |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| store.changeStoreData() |
| |
| } |
| |
| </script> |
| <template> |
| <h1>{{ msg }}</h1> |
| <h1>数量</h1> |
| <h2>pinia-count数据{{count}}</h2> |
| <h3>pinia- getters-2倍数量{{doubleCount}}</h3> |
| <h1>价格</h1> |
| <h2>pinia-price数据{{price}}</h2> |
| <h3>pinia- getters-2倍价格{{doublePrice}}</h3> |
| <el-button size='large' type="primary" @click="changeData">改变数据</el-button> |
| </template> |
| |
| <style scoped> |
| |
| </style> |
复制
4、actions(异步及调用其他actions方法)
新建store/use.ts模块
| import { defineStore } from 'pinia' |
| import { ElLoading } from 'element-plus' |
| type Res = { |
| nickname:string, |
| age:number |
| } |
| const Login = ()=>{ |
| return new Promise<Res>((resolve,reject)=>{ |
| const loading = ElLoading.service({ |
| lock: true, |
| text: 'Loading', |
| background: 'rgba(0, 0, 0, 0.7)', |
| }) |
| setTimeout(() => { |
| resolve({ |
| nickname:'张三', |
| age:22 |
| }) |
| loading.close() |
| }, 5000); |
| }) |
| } |
| export const userStore = defineStore('user',{ |
| state:()=>{ |
| return { |
| nickname:'', |
| age:0 |
| } |
| }, |
| getters:{ |
| |
| }, |
| actions:{ |
| async getUserInfo(){ |
| const res = await Login(); |
| this.nickname=res.nickname; |
| |
| |
| this.setAge(res.age) |
| }, |
| setAge(age:number){ |
| this.age=age |
| } |
| } |
| }) |
复制
| <template> |
| <h1>用户组件</h1> |
| <h2>昵称{{nickname}}--年龄{{age}}</h2> |
| <el-button type="primary" size="large" @click="getUserInfo">获取用户信息</el-button> |
| |
| </template> |
| <script setup lang='ts'> |
| import { ref,reactive} from 'vue' |
| import { userStore } from '../store/user'; |
| import { storeToRefs } from 'pinia'; |
| const user = userStore() |
| let {nickname,age} = storeToRefs(user) |
| const getUserInfo = ()=>{ |
| user.getUserInfo() |
| } |
| </script> |
| <style lang='scss' scoped> |
| </style> |
复制
5、getters (主要作用类似于computed 数据修饰并且有缓存)
getters 可以互相调用 普通函数形式可以使用this 使用箭头函数不能使用this this指向已经改变指向undefined 修改值请用state
| getters:{ |
| doubleCount():number{ |
| |
| return this.doublePrice*2 |
| }, |
| doublePrice:(state)=>state.price*2 |
| }, |
复制
6、 常用api
1、$reset(重置store
到他的初始状态)
| <el-button size='large' type="primary" @click="resetData">重置数据</el-button> |
| const resetData =()=>{ |
| main.$reset() |
| } |
复制
2、订阅state改变(类似于Vuex 的abscribe 只要有state 的变化就会走这个函数)
| main.$subscribe((args,state)=>{ |
| console.log(args,state,'数据改变') |
| }) |
复制

第二个参数
如果你的组件卸载之后还想继续调用请设置第二个参数
| main.$subscribe((args,state)=>{ |
| console.log(args,state,'数据改变') |
| },{ |
| detached:true |
| }) |
复制
3、订阅Actions的调用(只要有actions被调用就会走这个函数)
| main.$onAction((args)=>{ |
| console.log(args,'action调用'); |
| }) |
复制

7、 pinia插件 --数据持久化
1、安装
| cnpm i pinia-plugin-persist --save |
复制
2、新建store/index.ts
| import { createPinia } from 'pinia' |
| |
| import piniaPluginPersist from 'pinia-plugin-persist' |
| const store = createPinia() |
| store.use(piniaPluginPersist) |
| export default store |
| |
复制
3、修改main.ts
| import { createApp } from 'vue' |
| import App from './App.vue' |
| |
| import 'element-plus/dist/index.css' |
| import store from './store/index'; |
| import piniaPluginPersist from 'pinia-plugin-persist'; |
| store.use(piniaPluginPersist) |
| const app=createApp(App) |
| app.use(store).mount('#app') |
复制
4、保存nickname,修改store/user.ts
| import { defineStore } from 'pinia' |
| import { ElLoading } from 'element-plus' |
| type Res = { |
| nickname:string, |
| age:number |
| } |
| const Login = ()=>{ |
| return new Promise<Res>((resolve,reject)=>{ |
| const loading = ElLoading.service({ |
| lock: true, |
| text: 'Loading', |
| background: 'rgba(0, 0, 0, 0.7)', |
| }) |
| setTimeout(() => { |
| resolve({ |
| nickname:'张三', |
| age:22 |
| }) |
| loading.close() |
| }, 5000); |
| }) |
| } |
| export const userStore = defineStore('user',{ |
| state:()=>{ |
| return { |
| nickname:'', |
| age:0 |
| } |
| }, |
| getters:{ |
| |
| }, |
| actions:{ |
| async getUserInfo(){ |
| const res = await Login(); |
| this.nickname=res.nickname; |
| |
| |
| this.setAge(res.age) |
| }, |
| setAge(age:number){ |
| this.age=age |
| } |
| }, |
| persist:{ |
| enabled:true, |
| strategies:[ |
| { |
| storage:localStorage,paths:['nickname'] |
| } |
| ] |
| } |
| }) |
复制

4、购物车案例

1、新建src/api/shop.ts
| |
| |
| |
| |
| export interface IProduct { |
| id: number |
| title: string |
| price: number |
| inventory: number |
| } |
| |
| const _products: IProduct[] = [ |
| {id: 1, title: 'iPad 4 Mini', price: 500.01, inventory: 2}, |
| {id: 2, title: 'H&M T-Shirt White', price: 10.99, inventory: 10}, |
| {id: 3, title: 'Charli XCX -Sucker CD', price: 19.99, inventory: 5} |
| ] |
| |
| export const getProducts = async () => { |
| await wait(1000) |
| return _products |
| } |
| |
| export const buyProducts = async () => { |
| await wait(1000) |
| return Math.random() > 0.5 |
| } |
| |
| |
| |
| |
| |
| |
| async function wait(delay:number) { |
| return new Promise(resolve => setTimeout(resolve, delay)) |
| } |
复制
2、新建components/ProductList.vue
| <template> |
| <ul> |
| <li class="item" v-for="item in productsStore.all"> |
| <div class="title">商品名称:{{item.title}}</div>-- |
| <div class="price">商品价格:{{item.price}}</div> |
| <div class="price">商品库存:{{item.inventory}}</div> |
| <el-button type="primary" :disabled="!item.inventory" @click="cartStore.addProductToCart(item)">添加到购物车</el-button> |
| </li> |
| </ul> |
| </template> |
| |
| <script lang="ts" setup> |
| import { useCartStore } from '../store/cart'; |
| import { useProdunctsStore } from '../store/products' |
| |
| const productsStore = useProdunctsStore() |
| const cartStore = useCartStore() |
| |
| productsStore.loadAllProducts() |
| </script> |
| <style lang="scss" scoped> |
| .item{ |
| display: flex; |
| align-items: center; |
| justify-content: center; |
| .title{ |
| margin: 0 15px; |
| } |
| .price{ |
| margin: 0 15px; |
| } |
| } |
| </style> |
复制
3、新建components/Cart.vue
| <template> |
| <div class="cart"> |
| <h2>我的购物车</h2> |
| |
| <ul v-if="cartProducts&&cartProducts.length>0"> |
| <li v-for="item in cartProducts">商品名称:=:{{ item.title }} - 商品价格:{{ item.price }} × 商品数量:{{ item.num }}</li> |
| </ul> |
| <p v-else> |
| <i>请添加一些商品到购物车</i> |
| </p> |
| <p>商品总价: {{ totalPrice }}</p> |
| |
| </div> |
| </template> |
| |
| <script lang="ts" setup> |
| import { useCartStore } from '../store/cart' |
| import { storeToRefs } from 'pinia'; |
| const cartStore = useCartStore() |
| let { cartProducts,totalPrice } = storeToRefs(cartStore) |
| </script> |
复制
4、组件在App.vue中引用
| <script setup lang="ts"> |
| import ProductList from './components/ProductList.vue'; |
| import Cart from './components/Cart.vue'; |
| </script> |
| |
| <template> |
| <ProductList /> |
| <hr> |
| <Cart /> |
| </template> |
| |
| <style> |
| #app { |
| font-family: Avenir, Helvetica, Arial, sans-serif; |
| -webkit-font-smoothing: antialiased; |
| -moz-osx-font-smoothing: grayscale; |
| text-align: center; |
| color: #2c3e50; |
| margin-top: 60px; |
| } |
| ul li{ |
| list-style: none; |
| line-height: 50px; |
| } |
| </style> |
复制
5、新建store/products.ts
| import { defineStore } from "pinia" |
| import { getProducts, IProduct } from "../api/shop" |
| |
| export const useProdunctsStore = defineStore('products', { |
| state: () => { |
| return { |
| all: [] as IProduct[] |
| } |
| }, |
| |
| getters: {}, |
| |
| actions: { |
| |
| async loadAllProducts() { |
| const ret = await getProducts() |
| this.all = ret |
| }, |
| |
| decrementProduct(product: IProduct) { |
| const ret = this.all.find(item => item.id === product.id) |
| if (ret) { |
| ret.inventory-- |
| } |
| } |
| } |
| }) |
复制
6、新建store/cart.ts
| import { defineStore } from "pinia"; |
| import { buyProducts, IProduct } from "../api/shop"; |
| import { useProdunctsStore } from "./products"; |
| |
| |
| |
| |
| type CartProduct = { |
| num: number |
| } & Omit<IProduct, 'inventory'> |
| |
| export const useCartStore = defineStore('cart', { |
| state: () => { |
| return { |
| cartProducts: [] as CartProduct[], |
| checkoutStatus: null as null | string |
| } |
| }, |
| |
| getters: { |
| |
| totalPrice(state) { |
| return state.cartProducts.reduce((total, item) => { |
| return total + item.price * item.num |
| }, 0) |
| } |
| }, |
| |
| actions: { |
| |
| |
| |
| |
| |
| addProductToCart(product: IProduct) { |
| |
| if (product.inventory <= 0) { |
| return |
| } |
| |
| |
| const cartItem = this.cartProducts.find(item => item.id === product.id) |
| |
| if (cartItem) { |
| cartItem.num++ |
| } else { |
| |
| this.cartProducts.push({ |
| id: product.id, |
| title: product.title, |
| price: product.price, |
| num: 1 |
| }) |
| } |
| |
| |
| const productsStore = useProdunctsStore() |
| productsStore.decrementProduct(product) |
| |
| }, |
| |
| |
| |
| async checkOut() { |
| const ret = await buyProducts() |
| this.checkoutStatus = ret ? '成功' : '失败' |
| if (ret) { |
| this.cartProducts = [] |
| } |
| } |
| } |
| }) |
复制