watch 与 watchEffect 的区别

在开发 Vue 应用程序的整个过程中,您将拥有大量的响应性数据属性。您的应用程序将跟踪 input 字段、data 计算和一系列其他属性,并且可能需要在值更新时执行操作。

Vue 中的 watch 可以观察响应性属性,并可以检测到属性何时发生变化。它本质上充当组件响应式数据的事件监听器。

特别是当与异步 API 调用配合使用时,例如:

  • 当 ID 改变时,从数据库中获取一个对象
  • prop 更改时重新运行动画
  • 监听路由变化
  • input 输入框值的特殊处理等

在 Vue3 中,除了 watch 方法之外,还有一个新的 watchEffect 方法,可以在 Composition API 中使用。下面我们来简单介绍一下这两个方法。

如何使用 Vue watch?

Vue Options API 提供了一个 watch 选项,您可以在其中定义监视对象。要使用它,首先必须在 data 对象中具有要跟踪的属性。

export default {
  data () {
    return {
      title: 'Vue2'
    }
  },
  watch: {
    // Watcher
  }
}

然后,我们必须查看 watch 方法的结构。您所要做的就是声明一个与要观察的属性同名的函数。

它应该带有两个参数:

  1. 被监听属性的新 value
  2. 被监听属性的旧 value

例如,这是 title 属性的观察者。

watch: {
  title(newTitle, oldTitle) {
    console.log(`标题从 ${oldTitle} 更改为 ${newTitle})
  }
}

每当 title 值发生变化时,我们都可以在控制台中看到以下内容。

watch title

如果我们需要跟踪两个依赖项呢?在 Vue2 中,我们可以为创建两个观察者,但是 Vue3 Composition API 为该用例提供了更好的解决方案,它就是 watchEffect

Vue watchEffect 是如何工作的?

watchEffect 是 Vue3 中跟踪响应依赖关系的方法之一。本质上,我们可以使用响应性属性编写一个方法,每当它们的任何值更新时,我们的方法就会运行。

需要注意的一点是,除了在依赖项更改时运行外,watchEffect 还将在初始化组件时立即运行,因此在挂载 DOM 之前尝试访问 DOM 时要小心。

简单示例

让我们看一个非常简单的示例。

假设我们有某种响应性的 userID 属性,并且只想在它发生变化时打印出来。

import { ref, watchEffect } from 'vue'

export default {
  setup () {
    const userID = ref(0)

    watchEffect(() => console.log('Value: ' + userID.value))
    
    setTimeout(() => {
      userID.value = 1
    }, 1000)

    return {
      userID
    }
  }
}

当我们的组件初始化时,watchEffect 将运行并记录起始值。然后,每当 userID 的值改变时,就会触发 watchEffect

watchEffect

为什么与 watch 不同?

那么,既然已经有了 watch 方法,为什么这个新的 watchEffect 方法还会存在呢?

有一些关键的区别:

  • watchEffect 将在方法的任何依赖项发生更改时运行,watch 跟踪一个或多个特定的响应性属性,并且仅在该属性发生更改时运行。
  • 默认情况下,watch 是惰性的,因此仅当依赖项更改时才会触发。watchEffect 在创建组件后立即运行,然后跟踪依赖关系。

但是,我们可以为 watch 传递一个 immediate 属性,使其在初始化时运行。

export default {
  watch: {
    title: {
      handler: (newTitle, oldTitle) => {
        console.log("Title changed from " + oldTitle + " to " + newTitle);
      },
      immediate: true,
    },
  },
};

watch 在以下情况下很有用:

  • 您需要控制哪些依赖项会触发该方法
  • 您需要访问之前的值

这需要看您的项目适合哪一种。

对于 watcheffect 还需要注意的事

以下整理 Vue 文档对于 WatchEffect 需要注意的事。

props 是响应的

这是一个方便的小知识。一个很好的用例是根据 prop 属性更改页面上的其他值。

例如,假设我们有一个 SongPlayer 组件,该组件接受一个songID 值作为 prop。如果我们更改了歌曲,我们将希望加载有关它的所有内容。我们可以用 prop 来做到这一点。

export default {
  props: {
    songID: String,
  },
  setup(props) {
    watchEffect(() => console.log(props.songID))
  },
}

使用 onInvalidate() 清除副作用

使用 Vue watchEffect 的一个常见用例是,每当响应性依赖项发生变化时,就进行某种异步 API 调用。但是,如果依赖关系在第一个 API 调用完成之前再次发生变化,会发生什么呢?

这就是无效副作用出现的原因。

我们的 watchEffect 方法还有一个 onInvalidate 方法,每当该方法要再次运行或监视程序停止时,该方法就会运行。

我们可以像这样停止异步 API 调用。

export default {
  setup() {
    watchEffect(onInvalidate => {
      // 异步 API 调用
      const apiCall = someAsyncMethod(props.songID)

      onInvalidate(() => {
        // 取消 API 调用
        apiCall.cancel()
      })
    })
  }
}

停止观察者

我们还可以手动阻止一个观察者。如果我们想观察一个依赖项,直到它达到某个值,然后停止它,这可能很有用。如果我们在它达到目标值后继续观察,我们只是在浪费资源

我们的 Vue watchEffect 方法返回一个 stop 方法,该方法将停止我们的观察程序。

因此,我们可以简单地将其存储为变量,并在稍后调用它。

export default {
  setup (props) {
    let stopWatcher = watchEffect(() => console.log(props.songID))
    stopWatcher()
  }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,284评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,115评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,614评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,671评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,699评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,562评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,309评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,223评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,668评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,859评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,981评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,705评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,310评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,904评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,023评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,146评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,933评论 2 355