vue全家桶進階之路32:Vue3 WatchEffect和watch 監聽

来源:https://www.cnblogs.com/beichengshiqiao/archive/2023/04/17/17326349.html
-Advertisement-
Play Games

在 Vue 3 中,watchEffect 是一個用於監聽響應式數據變化的 API。它可以在函數內部自動跟蹤數據的依賴,併在依賴變化時重新運行函數。 watchEffect 的作用以及各個參數的功能講解: watchEffect(effect: (onInvalidate: InvalidateCb ...


在 Vue 3 中,watchEffect 是一個用於監聽響應式數據變化的 API。它可以在函數內部自動跟蹤數據的依賴,併在依賴變化時重新運行函數。

watchEffect 的作用以及各個參數的功能講解:

watchEffect(effect: (onInvalidate: InvalidateCbRegistrator) => void | (() => void) | Promise<void>, options?: WatchEffectOptions): WatchStopHandle

    • effect:要監聽的響應式數據以及需要執行的副作用函數。effect 函數接收一個 onInvalidate 回調函數作為參數,用於在 effect 的依賴發生變化時執行清理函數。effect 函數可以返回一個清理函數,也可以返回一個 Promise,Promise 完成時執行的函數也會被視為清理函數。如果 effect 函數中使用的數據沒有在響應式對象中被引用,那麼 watchEffect 不會監聽到這些數據的變化。
    • options:監聽選項對象,可選。可以用來配置監聽的行為,例如 deepflush 等選項。具體的選項請參考下麵的講解。
    • 返回值:一個函數,調用它可以停止監聽。

WatchEffectOptions

    watchEffect 函數接受一個可選的選項對象,用於配置監聽行為。以下是 WatchEffectOptions 對象的屬性及其功能:

    • lazy:是否延遲執行 effect 函數。如果設置為 true,則在首次調用 watchEffect 時不會執行 effect 函數,只有在其依賴發生變化時才會執行。預設為 false
    • deep:是否深度監聽對象和數組的變化,預設為 false。如果設置為 true,則會遞歸監聽對象和數組中的所有屬性。
    • flush:何時執行回調函數。預設為 'pre',表示在響應式數據變化後立即執行回調函數,但在同一事件迴圈中的其他變化不會觸發回調函數。如果設置為 'post',則會在同一事件迴圈中的所有變化都發生後執行回調函數。

例如,以下代碼使用 watchEffect 監聽 message 的變化,併在變化時執行回調函數。

import { watchEffect } from 'vue';

watchEffect(() => {
  console.log(`message 變為 ${message}`);
});

 

下麵是一個簡單的例子,演示瞭如何使用 watchEffect 監聽響應式數據變化:

<template>
  <div>
    <p>輸入框的值為:{{ message }}</p>
    <input v-model="message" />
  </div>
</template>

<script>
import { defineComponent, ref, watchEffect } from 'vue';

export default defineComponent({
  setup() {
    const message = ref('');

    watchEffect(() => {
      console.log(`message 值改變為:${message.value}`);
    });

    return {
      message
    };
  }
});
</script>

在這個例子中,我們使用 ref 函數定義了一個名為 message 的響應式數據,並將其初始化為空字元串。我們使用 watchEffect 函數監聽 message 的變化,併在函數內部列印出 message 的值。因為 watchEffect 函數會自動跟蹤 message 的依賴,所以當用戶在輸入框中輸入內容時,watchEffect 函數就會重新運行,列印出新的 message 值。

需要註意的是,watchEffect 函數的回調函數不需要顯式地指定依賴項,它會自動跟蹤回調函數內部使用到的所有響應式數據。因此,當你使用 watchEffect 函數時,不需要再使用 watch 函數或 computed 函數來監聽數據變化。但是,如果你需要監聽某個特定的響應式數據變化,可以在回調函數中使用該數據,這樣 watchEffect 就會自動跟蹤它。

watchEffect 函數的返回值是一個用於停止監聽的函數。當你調用這個函數時,watchEffect 就會停止監聽響應式數據的變化。

以下是一個示例,演示如何使用 watchEffect 函數的返回值停止監聽:

<template>
  <div>
    <p>輸入框的值為:{{ message }}</p>
    <button @click="stopWatching">停止監聽</button>
    <input v-model="message" />
  </div>
</template>

<script>
import { defineComponent, ref, watchEffect } from 'vue';

export default defineComponent({
  setup() {
    const message = ref('');

    const stopWatching = watchEffect(() => {
      console.log(`message 值改變為:${message.value}`);
    });

    function stopWatching() {
      stopWatching();
    }

    return {
      message,
      stopWatching
    };
  }
});
</script>

在這個例子中,我們定義了一個名為 stopWatching 的函數,它調用了 watchEffect 函數的返回值,從而停止了監聽。我們在模板中添加了一個按鈕,當用戶點擊它時,就會調用 stopWatching 函數,從而停止監聽 message 的變化。

 

 

watch的作用以及各個參數的功能講解:

watch 是 Vue 3 中用於監聽響應式數據變化的 API,它能夠在響應式數據發生變化時執行回調函數。以下是 watch 函數的參數及其功能:

  1. watch(source: string | Function | Ref, callback: Function, options?: WatchOptions): WatchStopHandle
  • source:要監聽的響應式數據,可以是一個字元串(表示要監聽的數據在組件實例中的屬性名)、一個函數(返回要監聽的數據)或一個 ref 對象。
  • callback:響應式數據變化時要執行的回調函數。回調函數接收兩個參數:新值和舊值。
  • options:監聽選項對象,可選。可以用來配置監聽的行為,例如 deepimmediate 等選項。具體的選項請參考下麵的講解。
  • 返回值:一個函數,調用它可以停止監聽。
  1. WatchOptions

watch 函數接受一個可選的選項對象,用於配置監聽行為。以下是 WatchOptions 對象的屬性及其功能:

  • immediate:是否在監聽開始時立即執行回調函數,預設為 false
  • deep:是否深度監聽對象和數組的變化,預設為 false。如果設置為 true,則會遞歸監聽對象和數組中的所有屬性。
  • flush:何時執行回調函數。預設為 'pre',表示在響應式數據變化後立即執行回調函數,但在同一事件迴圈中的其他變化不會觸發回調函數。如果設置為 'post',則會在同一事件迴圈中的所有變化都發生後執行回調函數。
  • onTrack:監視屬性被訪問的函數。它接收一個追蹤對象(TrackOpTypes)和追蹤的目標對象(target)。可以用它來進行依賴追蹤分析等操作。
  • onTrigger:監視屬性被修改的函數。它接收一個觸發對象(TriggerOpTypes)和觸發的目標對象(target)。可以用它來進行調試、性能分析等操作。
  • 例如,以下代碼監聽 message 的變化,併在變化時執行回調函數,同時設置 immediate 選項為 true,表示在組件掛載時立即執行一次回調函數。

import { watch } from 'vue';

watch(
  () => message, // 要監聽的響應式數據
  (newValue, oldValue) => {
    console.log(`message 從 ${oldValue} 變為 ${newValue}`);
  },
  {
    immediate: true // 在組件掛載時立即執行回調函數
  }
);

 

watchEffectwatch 都是 Vue 3 中用於監聽響應式數據變化的 API,它們之間的主要區別在於回調函數的類型和依賴項的聲明方式。

  • watchEffect 接收一個不帶參數的函數作為回調函數,在函數內部使用到的響應式數據變化時會自動觸發回調函數。watchEffect 會自動追蹤響應式數據的變化,併在每次變化時重新運行回調函數。
  • watch 接收兩個參數:第一個參數是要監聽的響應式數據,第二個參數是一個回調函數。回調函數接收兩個參數:新值和舊值。在響應式數據變化時,watch 會調用回調函數,並傳入新值和舊值。可以通過第三個參數來聲明要監聽的選項,例如 deepimmediate 等。

因為 watchEffect 自動追蹤響應式數據的變化,所以它更適合處理簡單的數據邏輯。如果需要監聽特定的響應式數據或使用更高級的選項,可以使用 watch 函數。使用 watch 函數需要顯式聲明要監聽的響應式數據,這可以使代碼更具可讀性。此外,watch 函數還支持聲明多個要監聽的響應式數據,這使得它可以處理更複雜的數據邏輯。

總之,如果你只需要監聽一些簡單的響應式數據變化,可以使用 watchEffect 函數。如果需要監聽特定的響應式數據或使用更高級的選項,可以使用 watch 函數。

以下是一個示例,演示 watchEffectwatch 的差別:

<template>
  <div>
    <p>message1 值為:{{ message1 }}</p>
    <p>message2 值為:{{ message2 }}</p>
    <button @click="increment">增加 message1 和 message2 的值</button>
  </div>
</template>

<script>
import { defineComponent, ref, watchEffect, watch } from 'vue';

export default defineComponent({
  setup() {
    const message1 = ref(0);
    const message2 = ref(0);

    // watchEffect 自動追蹤響應式數據的變化
    watchEffect(() => {
      console.log(`message1 值變為 ${message1.value}`);
    });

    // watch 需要顯式聲明要監聽的響應式數據
    watch(message2, (newValue, oldValue) => {
      console.log(`message2 從 ${oldValue} 變為 ${newValue}`);
    });

    function increment() {
      message1.value++;
      message2.value++;
    }

    return {
      message1,
      message2,
      increment
    };
  }
});
</script>

在這個例子中,我們聲明瞭兩個響應式數據 message1message2,並分別使用了 watchEffectwatch 監聽它們的變化。watchEffect 自動追蹤 message1 的變化,併在變化時觸發回調函數。而 watch 則需要顯式聲明要監聽的 message2,併在它的變化時觸發回調函數。在點擊按鈕時,我們同時增加了 message1message2 的值,從而觸發了相應的回調函數。

watchEffectwatch 的區別在於,watchEffect 監聽的是一個函數的副作用,而 watch 監聽的是一個具體的響應式數據,因此 watchEffect 不需要顯式指定監聽的數據,它會自動檢測 effect 函數中使用的響應式數據,併在其發生變化時執行回調函數。此外,watchEffect 也不需要手動停止監聽,它會在組件卸載時自動停止監聽。但是,watchEffect 不支持監聽選項對象中的 immediateonTrackonTrigger 屬性。

 

 

如果watch的監聽對象是數組:

如果 watch 監聽的是一個數組,則可以使用 deep 選項來深度監聽數組元素的變化。當 deeptrue 時,watch 會遞歸監聽數組中每個元素的變化。

例如,以下代碼使用 watch 監聽 list 數組的變化,併在變化時執行回調函數。

import { watch } from 'vue';

watch(
  () => list,
  (newList, oldList) => {
    console.log('list 變化了', newList, oldList);
  },
  { deep: true }
);

在上面的代碼中,watch 監聽的是一個計算屬性,計算屬性返回 list 數組。由於在監聽選項中設置了 deep: true,因此 watch 會深度監聽 list 數組,即遞歸監聽數組中每個元素的變化。

註意,當使用 deep 選項監聽數組時,如果數組中的元素是對象,則需要確保這些對象是響應式的,否則無法監聽它們的變化。如果數組中的元素不是響應式的對象,則無法監聽其變化。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 如今,數字經濟正逐步走向深化應用、規範發展、普惠共用的新階段,數字經濟與實體經濟深度融合、基礎軟體國產化替代成為數字時代主潮流。數字工具如何讓千行百業共同實現韌性生長? 「 2023 袋鼠雲春季生長大會」乘風而起,帶來數實融合趨勢下的產品煥新升級剖析、前瞻行業視覺解讀、最佳數字實踐分享,助力各大產業 ...
  • 摘要:今天發現Mysql的主從資料庫沒有同步,瞬間整個人頭皮發麻。 本文分享自華為雲社區《糟了,生產環境數據竟然不一致,人麻了!》,作者:冰 河 。 今天發現Mysql的主從資料庫沒有同步 先上Master庫: mysql>show processlist; 查看下進程是否Sleep太多。發現很正常 ...
  • GreatSQL社區原創內容未經授權不得隨意使用,轉載請聯繫小編並註明來源。 GreatSQL是MySQL的國產分支版本,使用上與MySQL一致。 作者: 奧特曼愛小怪獸 文章來源:GreatSQL社區投稿 上一篇 MySQL8.0 優化器介紹(一)介紹了成本優化模型的三要素:表關聯順序,與每張表返 ...
  • 一、資料庫類型 關係資料庫管理系統(RDBMS) 非關係資料庫管理系統(NoSQL) 按照預先設置的組織機構,將數據存儲在物理介質上(即:硬碟上) 數據之間可以做無關聯操作 (例如: 多表查詢,嵌套查詢,外鍵等) 主流的RDBMS軟體:MySQL、MariaDB、Oracle、DB2、SQL Ser ...
  • TiDB作為NewSQL,其在對MySQL(SQL92協議)的相容上做了很多,MySQL作為當下使用較廣的事務型資料庫,在IT界尤其是互聯網間使用廣泛,那麼對於開發人員來說,1)兩個資料庫產品在SQL開發及調優的過程中,都有哪些差異?在系統遷移前需要提前做哪些準備? 2)TiDB的執行計劃如何查看,... ...
  • GreatSQL社區原創內容未經授權不得隨意使用,轉載請聯繫小編並註明來源。 GreatSQL是MySQL的國產分支版本,使用上與MySQL一致。 作者: Yejinrong/葉金榮 文章來源:GreatSQL社區投稿 啟用coredump 製造一個coredump場景 真實故障場景分析跟蹤 上一篇 ...
  • 背景 如今很多網站都引入截圖功能,可用於問題反饋、內容分享等實用需求,而前端截圖也不知不覺成為了首選。今天為大家推薦兩種前端截圖方式,雖然有些局限,但是也能應付大部分項目需求。 Canvas截圖:html2canvas SVG截圖:rasterizehtml 原理 首先來談下兩種前端截圖方式的原理, ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 隔行換色(%): window.onload = function() { var aLi = document.getElementsByTagName('li'); for(var i = 0; i < aLi.length; i++ ...
一周排行
    -Advertisement-
    Play Games
  • 前言 本文介紹一款使用 C# 與 WPF 開發的音頻播放器,其界面簡潔大方,操作體驗流暢。該播放器支持多種音頻格式(如 MP4、WMA、OGG、FLAC 等),並具備標記、實時歌詞顯示等功能。 另外,還支持換膚及多語言(中英文)切換。核心音頻處理採用 FFmpeg 組件,獲得了廣泛認可,目前 Git ...
  • OAuth2.0授權驗證-gitee授權碼模式 本文主要介紹如何筆者自己是如何使用gitee提供的OAuth2.0協議完成授權驗證並登錄到自己的系統,完整模式如圖 1、創建應用 打開gitee個人中心->第三方應用->創建應用 創建應用後在我的應用界面,查看已創建應用的Client ID和Clien ...
  • 解決了這個問題:《winForm下,fastReport.net 從.net framework 升級到.net5遇到的錯誤“Operation is not supported on this platform.”》 本文內容轉載自:https://www.fcnsoft.com/Home/Sho ...
  • 國內文章 WPF 從裸 Win 32 的 WM_Pointer 消息獲取觸摸點繪製筆跡 https://www.cnblogs.com/lindexi/p/18390983 本文將告訴大家如何在 WPF 裡面,接收裸 Win 32 的 WM_Pointer 消息,從消息裡面獲取觸摸點信息,使用觸摸點 ...
  • 前言 給大家推薦一個專為新零售快消行業打造了一套高效的進銷存管理系統。 系統不僅具備強大的庫存管理功能,還集成了高性能的輕量級 POS 解決方案,確保頁面載入速度極快,提供良好的用戶體驗。 項目介紹 Dorisoy.POS 是一款基於 .NET 7 和 Angular 4 開發的新零售快消進銷存管理 ...
  • ABP CLI常用的代碼分享 一、確保環境配置正確 安裝.NET CLI: ABP CLI是基於.NET Core或.NET 5/6/7等更高版本構建的,因此首先需要在你的開發環境中安裝.NET CLI。這可以通過訪問Microsoft官網下載並安裝相應版本的.NET SDK來實現。 安裝ABP ...
  • 問題 問題是這樣的:第三方的webapi,需要先調用登陸介面獲取Cookie,訪問其它介面時攜帶Cookie信息。 但使用HttpClient類調用登陸介面,返回的Headers中沒有找到Cookie信息。 分析 首先,使用Postman測試該登陸介面,正常返回Cookie信息,說明是HttpCli ...
  • 國內文章 關於.NET在中國為什麼工資低的分析 https://www.cnblogs.com/thinkingmore/p/18406244 .NET在中國開發者的薪資偏低,主要因市場需求、技術棧選擇和企業文化等因素所致。歷史上,.NET曾因微軟的閉源策略發展受限,儘管後來推出了跨平臺的.NET ...
  • 在WPF開發應用中,動畫不僅可以引起用戶的註意與興趣,而且還使軟體更加便於使用。前面幾篇文章講解了畫筆(Brush),形狀(Shape),幾何圖形(Geometry),變換(Transform)等相關內容,今天繼續講解動畫相關內容和知識點,僅供學習分享使用,如有不足之處,還請指正。 ...
  • 什麼是委托? 委托可以說是把一個方法代入另一個方法執行,相當於指向函數的指針;事件就相當於保存委托的數組; 1.實例化委托的方式: 方式1:通過new創建實例: public delegate void ShowDelegate(); 或者 public delegate string ShowDe ...