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
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...