Vue3 響應(yīng)式偵聽與計算的實現(xiàn)
響應(yīng)式偵聽和計算
有時我們需要依賴于其他狀態(tài)的狀態(tài)——在 Vue 中,這是用組件 計算屬性 處理的,以直接創(chuàng)建計算值,我們可以使用 computed 方法:它接受 getter 函數(shù)并為 getter 返回的值返回一個不可變的響應(yīng)式 ref 對象。
我們先來看看一個簡單的例子,關(guān)于計算值的方式,同樣我們在 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)式追蹤其依賴,并在其依賴變更時重新運行該函數(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>
查看效果如下:

停止偵聽
當 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>
查看效果如下:

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

