Vue3 響應(yīng)式偵聽與計(jì)算的實(shí)現(xiàn)
響應(yīng)式偵聽和計(jì)算
有時我們需要依賴于其他狀態(tài)的狀態(tài)——在 Vue 中,這是用組件 計(jì)算屬性 處理的,以直接創(chuàng)建計(jì)算值,我們可以使用 computed
方法:它接受 getter 函數(shù)并為 getter 返回的值返回一個不可變的響應(yīng)式 ref 對象。
我們先來看看一個簡單的例子,關(guān)于計(jì)算值的方式,同樣我們在 src/TemplateM.vue
寫下如下代碼:
<template> <div class="template-m-wrap"> count ---> {{count}} plusOne ---> {{plusOne}} </div> </template> <script> import { ref, defineComponent, reactive, computed } from "vue"; export default defineComponent({ name: 'TemplateM', setup() { let count = ref(2) let plusOne = computed(() => { return count.value++ }) console.log(plusOne.value) return { count, plusOne } } }) </script>
訪問鏈接效果如下:
我們可以看到 plusOne
沒有值?;蛘?,它可以使用一個帶有 get
和 set
函數(shù)的對象來創(chuàng)建一個可寫的 ref 對象。
<template> <div class="template-m-wrap"> count ---> {{ count }} plusOne ---> {{ plusOne }} </div> </template> <script> import { ref, defineComponent, reactive, computed } from "vue"; export default defineComponent({ name: "TemplateM", setup() { let count = ref(2); let plusOne = computed({ get() { return count.value++; }, set(val) { count.value = val; }, }); plusOne.value = 1; console.log(count.value); // 0 return { count, plusOne, }; }, }); </script>
同樣訪問效果如下:
watchEffect
為了根據(jù)響應(yīng)式狀態(tài) 自動應(yīng)用 和 重新應(yīng)用 副作用,我們可以使用 watchEffect
方法。它立即執(zhí)行傳入的一個函數(shù),同時響應(yīng)式追蹤其依賴,并在其依賴變更時重新運(yùn)行該函數(shù)。
<template> <div class="template-m-wrap"> count ---> {{ count }} </div> </template> <script> import { ref, defineComponent, reactive, computed, watchEffect } from "vue"; export default defineComponent({ name: "TemplateM", setup() { let count = ref(2); watchEffect(() => { console.log(count.value) }) setTimeout(() => { count.value++ }, 1000) return { count, }; }, }); </script>
查看效果如下:
停止偵聽
當(dāng) watchEffect
在組件的 setup() 函數(shù)或 生命周期鉤子 被調(diào)用時,偵聽器會被鏈接到該組件的生命周期,并在組件卸載時自動停止。
在一些情況下,也可以顯式調(diào)用返回值以停止偵聽:
<template> <div class="template-m-wrap"> count ---> {{ count }} </div> </template> <script> import { ref, defineComponent, reactive, computed, watchEffect } from "vue"; export default defineComponent({ name: "TemplateM", setup() { let count = ref(2); const stopWatch = watchEffect(() => { console.log(count.value) }) stopWatch() setTimeout(() => { count.value++ }, 1000) return { count, }; }, }); </script>
查看效果如下:
副作用刷新時機(jī)
Vue 的響應(yīng)性系統(tǒng)會緩存副作用函數(shù),并異步地刷新它們,這樣可以避免同一個“tick” 中多個狀態(tài)改變導(dǎo)致的不必要的重復(fù)調(diào)用。在核心的具體實(shí)現(xiàn)中,組件的 update
函數(shù)也是一個被偵聽的副作用。當(dāng)一個用戶定義的副作用函數(shù)進(jìn)入隊(duì)列時,默認(rèn)情況下,會在所有的組件 update
前 執(zhí)行:
<template> <div>{{ count }}</div> </template> <script> export default { setup() { const count = ref(0) watchEffect(() => { console.log(count.value) }) return { count } } } </script>
在這個例子中:
count
會在初始運(yùn)行時同步打印出來- 更改
count
時,將在組件 更新前 執(zhí)行副作用。
如果需要在組件更新 后 重新運(yùn)行偵聽器副作用,我們可以傳遞帶有 flush
選項(xiàng)的附加 options
對象 (默認(rèn)為 'pre'
):
// fire before component updates watchEffect( () => { /* ... */ }, { flush: 'post' } )
flush
選項(xiàng)還接受 sync
,這將強(qiáng)制效果始終同步觸發(fā)。然而,這是低效的,應(yīng)該很少需要。
偵聽器調(diào)試
onTrack
和 onTrigger
選項(xiàng)可用于調(diào)試偵聽器的行為。
- onTrack 將在響應(yīng)式 property 或 ref 作為依賴項(xiàng)被追蹤時被調(diào)用。
- onTrigger 將在依賴項(xiàng)變更導(dǎo)致副作用被觸發(fā)時被調(diào)用。
這兩個回調(diào)都將接收到一個包含有關(guān)所依賴項(xiàng)信息的調(diào)試器事件。建議在以下回調(diào)中編寫 debugger
語句來檢查依賴關(guān)系:
watchEffect( () => { /* 副作用 */ }, { onTrigger(e) { debugger } } )
onTrack
和 onTrigger
只能在開發(fā)模式下工作。
watch
watch
API 完全等同于組件偵聽器 property。 watch
需要偵聽特定的數(shù)據(jù)源,并在回調(diào)函數(shù)中執(zhí)行副作用。默認(rèn)情況下,它也是惰性的,即只有當(dāng)被偵聽的源發(fā)生變化時才執(zhí)行回調(diào)。
與watchEffect 比較, watch
允許我們:
- 懶執(zhí)行副作用;
- 更具體地說明什么狀態(tài)應(yīng)該觸發(fā)偵聽器重新運(yùn)行;
- 訪問偵聽狀態(tài)變化前后的值。
偵聽單個數(shù)據(jù)源
偵聽器數(shù)據(jù)源可以是返回值的 getter 函數(shù),也可以直接是 ref
:
// 偵聽一個 getter const state = reactive({ count: 0 }) watch( () => state.count, (count, prevCount) => { /* ... */ } ) // 直接偵聽ref const count = ref(0) watch(count, (count, prevCount) => { /* ... */ })
到此這篇關(guān)于Vue3 響應(yīng)式偵聽與計(jì)算的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Vue3 響應(yīng)式偵聽與計(jì)算內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Vuex unknown action type報錯問題及解決
這篇文章主要介紹了Vuex unknown action type報錯問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-02-02vue項(xiàng)目引入本地bootstrap不生效問題及解決
這篇文章主要介紹了vue項(xiàng)目引入本地bootstrap不生效問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-10-10Vue2.0基于vue-cli+webpack Vuex的用法(實(shí)例講解)
下面小編就為大家?guī)硪黄猇ue2.0基于vue-cli+webpack Vuex的用法(實(shí)例講解)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-09-09vue 數(shù)組和對象不能直接賦值情況和解決方法(推薦)
這篇文章主要介紹了vue 數(shù)組和對象不能直接賦值情況和解決方法,需要的朋友可以參考下2017-10-10使用vue-draggable-plus實(shí)現(xiàn)拖拽排序
最近遇到一個需求,在 Vue3 的一個 H5 頁面當(dāng)中點(diǎn)擊拖拽圖標(biāo)上下拖動 tab 子項(xiàng),然后點(diǎn)擊保存可以保存最新的 tab 項(xiàng)順序,同事說可以用 vue-draggable-plus 這個庫來實(shí)現(xiàn)拖拽,所以本文給大家介紹了如何使用vue-draggable-plus實(shí)現(xiàn)拖拽排序,需要的朋友可以參考下2024-01-01Element Table 自適應(yīng)高度的實(shí)現(xiàn)示例
el-table的高度不能適應(yīng)不同電腦的分辨率,也不能跟隨瀏覽器的高度變化而變化的問題,本文就來解決一下Element Table 自適應(yīng)高度,感興趣的可以了解一下2024-07-07vue elementui el-table 表格里邊展示四分位圖效果(功能實(shí)現(xiàn))
這篇文章主要介紹了vue elementui el-table 表格里邊展示四分位圖效果(功能實(shí)現(xiàn)),本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-04-04proxy代理不生效以及vue?config.js不生效解決方法
在開發(fā)Vue項(xiàng)目過程中,使用了Proxy代理進(jìn)行數(shù)據(jù)劫持,但是在實(shí)際運(yùn)行過程中發(fā)現(xiàn)代理并沒有生效,也就是說數(shù)據(jù)并沒有被劫持,這篇文章主要給大家介紹了關(guān)于proxy代理不生效以及vue?config.js不生效解決方法的相關(guān)資料,需要的朋友可以參考下2023-11-11