溫馨提示:本文以vue3+vite+ts舉例,vite配置和ts語法側重較少,比較適合有vuex或者vue基礎的小伙伴們兒查閱。 安裝pinia yarn yarn add pinia npm npm install pinia pnpm pnpm add pinia 1-開始 方式一:在main. ...
溫馨提示:本文以vue3+vite+ts舉例,vite配置和ts語法側重較少,比較適合有vuex或者vue基礎的小伙伴們兒查閱。
安裝pinia
- yarn
yarn add pinia
- npm
npm install pinia
- pnpm
pnpm add pinia
1-開始
方式一:在main.ts
中直接引入pinia
在 src/main.ts
中引入pinia(根存儲),並傳遞給應用程式。
import { createApp } from 'vue'
import './style.css'
import App from './App.vue'
// 1-創建一個 pinia(根存儲)
import { createPinia } from 'pinia'
const app = createApp(App)
// 2-告訴應用程式,我們將使用pinia
const pinia = createPinia();
// 以插件形式傳遞給app
app.use(pinia);
app.mount('#app');
方式二(推薦):單獨開個.ts
文件引入pinia
在根目錄下新建文件夾,這裡我命名為store
,再在文件夾下新建一個index.ts
文件(src/store/index.ts
),用以配置和引入pinia。
// 1-創建一個 pinia(根存儲)
import { createPinia } from 'pinia'
// 2-定義pinia實例
const pinia = createPinia();
// 3-暴露pinia實例
export default pinia;
然後在src/main.ts
中使用。
......
import pinia from '@/store/index.ts'
app.use(pinia);
......
其實和方式一沒啥區別,只是為了保持main.ts
文件整潔,並且方便配置pinia。
2-創建倉庫
pinia與vuex差不多,相比於vuex,少了mutation
和modules
。
pinia創建倉庫,有選項式寫法和組合式寫法。
選項式Options API寫法
在根目錄下創建一個文件夾store (src/store
),在store文件夾中可以創建你的倉庫,比如下麵我創建了一個名為user的倉庫 (src/store/user.ts
)。
// 選項式寫法
// 1-引入api
import { defineStore } from "pinia";
// 2-定義倉庫
const store = defineStore('user', {
// 3-設置組件共用的狀態,相當於組件的data
state: () => ({
userInfo: {
name: '老劉',
sex: '男',
age: 17,
isStudent: false
},
token: '5201314',
password: '123456',
}),
// 3-設置狀態計算值,相當於組件的computed
getters: {
name: (state) => state.userInfo.name,
sex: (state) => state.userInfo.sex,
},
// 3-設置組件共用的方法,相當於組件的methods
actions: {
addAge() {
this.userInfo.age++;
}
}
});
// 最後別忘了把倉庫暴露出去哦
export default store;
組合式Composition API寫法(推薦)
上面的倉庫 (src/store/user.ts
)組合式寫法如下:
// 組合式寫法
// 1-引入pinia的api
import { defineStore } from "pinia";
// 2-引入vue3相關api
import { ref, reactive, computed } from 'vue';
// 3-定義倉庫,註意第二個參數需要傳入一個函數,函數需要返回一個對象!
const store = defineStore('user', () => {
// 4-在這裡面可以像在組件中一樣,使用vue3的API,定義響應式數據
const userInfo = reactive({
name: '老劉',
sex: '男',
age: 17,
isStudent: false
});
const token = ref('5201314');
const password = ref('123456');
// 這裡computed的作用相當於getters
const name = computed(() => userInfo.name);
const sex = computed(() => userInfo.sex);
// 4-還可以定義方法
function addAge() {
userInfo.age++;
}
// 5-然後把需要共用的數據或方法,裝進一個對象,return出去
return {
userInfo,
token,
password,
name,
sex,
addAge
}
});
// 最後別忘了把倉庫暴露出去哦
export default store;
TIP
還可以在倉庫中使用
watch
、watchEffect
等vue3的API喔~。import { ref, reactive, computed, watch } from 'vue'; const store = defineStore('user', () => { const userInfo = reactive({ age: 17, }); // 使用vue3的watch()函數,可以對倉庫狀態進行監聽 watch(() => userInfo.age, (val) => { console.log(val); }); });
3-使用pinia
完成了上面的工作後,我們就可以在組件中愉快地使用pinia啦!
下麵以src/App.vue
作為示例。
(1)引入倉庫
<template>
</template>
<script setup lang="ts">
// 1-引入剛剛自定義的倉庫,模塊名store 可以自定義
import store from '@/store/user.ts';
// 2-使用倉庫,倉庫實例名userStore 可以自定義
const userStore = store();
</script>
(2)在組件中訪問倉庫state
和getters
在模板和script中,state和getters可以看作倉庫實例(如userStore
)的屬性,直接加.
訪問即可。
<template>
<div>
<!-- 1-訪問倉庫的計算屬性getters -->
<span>姓名:{{ userStore.name }}</span>
<span>性別:{{ userStore.sex }}</span>
<!-- 1-訪問倉庫的狀態state -->
<span>年齡:{{ userStore.userInfo.age }}</span>
<span>是否學生:{{ userStore.userInfo.isStudent ? '是' : '否' }}</span>
</div>
</template>
<script setup lang="ts">
import store from '@/store/user.ts';
const userStore = store();
// 1-訪問state和getters,類似於reactive響應式數據的訪問
console.log('userStore', userStore);
console.log('姓名:', userStore.name);
console.log('性別:', userStore.sex);
console.log('年齡:', userStore.userInfo.age);
console.log('是否學生:', userStore.userInfo.isStudent ? '是' : '否');
</script>
輸出
userStore Proxy(Object) {$id: 'user', $onAction: ƒ, $patch: ƒ, $reset: ƒ, $subscribe: ƒ, …}[[Handler]]: Object[[Target]]: Object[[IsRevoked]]: false
姓名: 老劉
性別: 男
年齡: 17
是否學生: 否
(3)在組件中使用倉庫actions
使用倉庫方法與訪問倉庫state類似,倉庫實例後直接加.
調用即可。
<template>
<div>
<!-- 按鈕點擊,年齡+1 -->
<button @click="onAddAge">年齡+1</button>
<span>年齡:{{ userStore.userInfo.age }}</span>
</div>
</template>
<script setup lang="ts">
import store from '@/store/user.ts';
const userStore = store();
// 按鈕點擊觸發
function onAddAge() {
// 1-使用倉庫的actions
userStore.addAge();
}
</script>
(4)修改state
直接修改
與vuex不同,pinia支持在組件中直接修改state
。
<template>
<div>
<!-- 按鈕點擊,年齡+1 -->
<button @click="onAddAge">年齡+1</button>
<span>年齡:{{ userStore.userInfo.age }}</span>
</div>
</template>
<script setup lang="ts">
import store from '@/store/user.ts';
const userStore = store();
// 按鈕點擊觸發
function onAddAge() {
// 1-直接修改state
userStore.userInfo.age++;
}
</script>
利用倉庫actions
進行修改
src/store/user.ts
......
const store = defineStore('user', () => {
......
// 1-定義方法
function addAge() {
userInfo.age++;
}
// 2-return出去
return {
addAge
}
});
// 3-導出倉庫
export default store;
src/App.vue
<template>
<div>
<!-- 按鈕點擊,年齡+1 -->
<button @click="onAddAge">年齡+1</button>
<span>年齡:{{ userStore.userInfo.age }}</span>
</div>
</template>
<script setup lang="ts">
import store from '@/store/user.ts';
const userStore = store();
// 按鈕點擊觸發
function onAddAge() {
// 4-使用倉庫的方法
userStore.addAge();
}
</script>
批量變更
通過倉庫實例(如userStore
)的 $patch
方法,可以對state
同時應用多個更改。
<script setup lang="ts">
......
function changeState() {
console.log(userStore.token); // '5201314'
console.log(userStore.password); // '123456'
// $patch()接收一個對象,對象內的屬性是 需要變更的state
// 註意是變更,新增state是無效的!
userStore.$patch({
token: '1024',
password: '654321'
});
console.log(userStore.token); // '1024'
console.log(userStore.password); // '654321'
}
</script>
上面的方法每次進行批量修改都需要傳入一個新對象,有時候使用起來並不方便。下麵是另一種寫法,$patch
接受一個函數來批量修改集合內部分對象。(推薦)
<script setup lang="ts">
......
function changeState() {
// 這裡的any是typescript的類型標註,可以不用理會
// 回調函數的參數state就是 倉庫目前的state
userStore.$patch((state: any) => {
state.token = '1024';
state.password = '654321';
});
}
</script>
整體替換(不推薦)
通過倉庫實例(如userStore
)的 $state
屬性,來為新對象替換倉庫的整個狀態。
pinia官網提到整體替換state的方法,但並未說明是否保留數據響應式。經筆者實踐,這種方法會丟失數據的響應式,所以不推薦使用。
<script setup lang="ts">
......
function updateStore() {
userStore.$state = {
userInfo: {
name: '老王',
sex: '男',
age: 66,
isStudent: false
}
};
}
</script>
(5)重置state
通過調用倉庫實例上的 $reset()
方法將狀態重置到其初始值。
<script setup lang="ts">
......
function resetStore() {
userStore.$reset();
}
</script>
$reset()
的坑
細心的你會發現,倉庫state
並沒有重置,然後你打開你的的控制台,你會驚訝地發現它報了這麼一個錯誤:
這時候請你不要慌,先冷靜地看一下報錯信息。
這裡翻譯一下:Store "user"是使用setup語法構建的,不實現$reset()
。(猜測是pinia的缺陷)
所以,根據報錯信息,這裡提供下麵兩種解決方案。
使用選項式Options API
使用選項式Options API編寫pinia倉庫,並且在組件中不能用script setup語法,要使用setup函數進行開發。
src/store/user.ts
......
// 使用選項式Options API編寫倉庫
const store = defineStore('user', {
// 3-設置組件共用的狀態,相當於組件的data
state: () => ({
userInfo: {
name: '老劉',
sex: '男',
age: 17,
isStudent: false
},
token: '5201314',
password: '123456',
}),
});
export default store;
src/App.vue
<!-- 這裡不能用script setup,否則依然報錯 -->
<script lang="ts">
import store from '@/store/user.ts';
export default {
setup() {
const userStore = store();
function resetStore() {
// 重置state
userStore.$reset();
}
// 把響應式數據或方法return出去
return {
userStore,
resetStore
}
},
}
</script>
重寫一個$reset()
方法(推薦)
原理:自定義pinia插件(Plugins),利用$patch()
重置整個state
。
在之前創建的pinia配置文件中修改(src/store/index.ts
)。
import { createPinia } from 'pinia';
const pinia = createPinia();
// 1-使用pinia自定義插件
pinia.use(({ store }) => {
// 2-獲取最開始的State
const initialState = JSON.parse(JSON.stringify(store.$state));
// 3-重寫$reset()方法
store.$reset = () => {
// 4-利用$patch()批量變更state,達到重置state的目的
store.$patch(initialState);
}
});
export default pinia;
推薦使用這種方法,這樣就可以在script setup
中愉快地使用pinia啦!
完整示例
倉庫配置
src/store/index.ts
import { createPinia } from 'pinia';
const pinia = createPinia();
pinia.use(({ store }) => {
const initialState = JSON.parse(JSON.stringify(store.$state));
store.$reset = () => {
store.$patch(initialState);
}
});
export default pinia;
定義倉庫
src/store/user.ts
// 組合式寫法
import { defineStore } from "pinia";
import { ref, reactive, computed, watch } from 'vue';
const store = defineStore('user', () => {
const userInfo = reactive({
name: '老劉',
sex: '男',
age: 17,
isStudent: false
});
const token = ref('5201314');
const password = ref('123456');
const name = computed(() => userInfo.name);
const sex = computed(() => userInfo.sex);
watch(() => userInfo.age, (val) => {
console.log(val);
});
function addAge() {
userInfo.age++;
}
return {
userInfo,
token,
password,
name,
sex,
addAge
}
});
export default store;
父組件
src/App.vue
<template>
<div>
<button @click="resetStore">重置store</button>
<h1>我是父組件</h1>
<button @click="userStore.userInfo.age++">年齡+1</button>
<span class="marginLeft60">姓名:{{ userStore.name }}</span>
<span class="marginLeft60">性別:{{ userStore.sex }}</span>
<span class="marginLeft60 red">年齡:{{ userStore.userInfo.age }}</span>
<span class="marginLeft60 red">是否學生:{{ userStore.userInfo.isStudent ? '是' : '否' }}</span>
<hr>
<!-- 使用子組件A -->
<son-a />
<hr>
<!-- 使用子組件B -->
<son-b />
<hr>
</div>
</template>
<script setup lang="ts">
// 引入子組件
import SonA from './components/sonA.vue';
import SonB from './components/sonB.vue';
// 1-引入倉庫
import store from '@/store/user.ts';
// 2-使用倉庫
const userStore = store();
// 重置store
function resetStore() {
userStore.$reset();
}
</script>
<style>
.marginLeft60 {
margin-left: 60px;
}
.red {
color: red;
}
</style>
子組件A
src/components/sonA.vue
<template>
<div>
<h2>我是子組件A</h2>
<button @click="userStore.userInfo.isStudent = true">入學</button>
<span class="marginLeft60">姓名:{{ userStore.name }}</span>
<span class="marginLeft60">性別:{{ userStore.sex }}</span>
<span class="marginLeft60 red">年齡:{{ userStore.userInfo.age }}</span>
<span class="marginLeft60 red">是否學生:{{ userStore.userInfo.isStudent ? '是' : '否' }}</span>
<grandson />
</div>
</template>
<script setup lang="ts">
import Grandson from './grandson.vue';
import store from '@/store/user.ts';
const userStore = store();
</script>
子組件B
src/components/sonB.vue
<template>
<div>
<h2>我是子組件B</h2>
<button @click="userStore.userInfo.isStudent = false">畢業</button>
<span class="marginLeft60">姓名:{{ userStore.name }}</span>
<span class="marginLeft60">性別:{{ userStore.sex }}</span>
<span class="marginLeft60 red">年齡:{{ userStore.userInfo.age }}</span>
<span class="marginLeft60 red">是否學生:{{ userStore.userInfo.isStudent ? '是' : '否' }}</span>
</div>
</template>
<script setup lang="ts">
import store from '@/store/user.ts';
const userStore = store();
</script>
孫組件
src/components/grandson.vue
<template>
<div>
<h3>我是孫組件</h3>
<span class="marginLeft60">姓名:{{ userStore.name }}</span>
<span class="marginLeft60">性別:{{ userStore.sex }}</span>
<span class="marginLeft60 red">年齡:{{ userStore.userInfo.age }}</span>
<span class="marginLeft60 red">是否學生:{{ userStore.userInfo.isStudent ? '是' : '否' }}</span>
</div>
</template>
<script setup lang="ts">
import store from '@/store/user.ts';
const userStore = store();
</script>