computed 是模板表达式的声明式描述,会创建新的响应式数据。而 watch 是响应式数据的 自定义侦听器,用于响应数据的变化。除此之外,computed 还具有可缓存,可依赖多个属 性,getter 函数无副作用等特点。watch 则更适用于异步或开销大的操作。
1. 实现原理
在了解 Vue 数据双向绑定的基础上,computed 等同于为属性设置 getter 函数 (也可设置 setter),而 watch 等同于为属性的 setter 设置回调函数、监听深度 deep 及响应速度 immediate。下面简单讲解下两者的实现原理,具体细节可以参考源码。
1.1 computed 原理
主要分为四个阶段
- 初始化:为 computed 属性创建 lazy watcher(此处 watcher 指双向绑定中的监听 器,下同)。
- 首次模板渲染:渲染 watcher 检测到 computed 属性时,会调用 computed 属性的 getter 方法,而 computed 属性的 getter 方法会调用依赖属性的 getter,从而形成链 式调用,同时保存引用关系用于更新。取得计算结果后 lazy watcher 会将结果缓存,并 返回给渲染 watcher 进行模板渲染。
- 多次模板渲染:直接取 lazy watcher 中的缓存值给到渲染 watcher 进行渲染。
- 依赖属性更新:根据首次模板渲染阶段构建的依赖关系向上通知 lazy watcher 进行重新 计算,缓存计算结果并通知渲染 watcher 重新渲染更新页面。
1.2 watch 原理
watch 本质上是为每个监听属性 setter 创建了一个 watcher,当被监听的属性更新时,调用 传入的回调函数。常见的配置选项有 deep 和 immediate,对应原理如下:
- deep:深度监听对象,为对象的每一个属性创建一个 watcher,从而确保对象的每一个 属性更新时都会触发传入的回调函数。主要原因在于对象属于引用类型,单个属性的更 新并不会触发对象 setter,因此引入 deep 能够很好地解决监听对象的问题。同时也会 引入判断机制,确保在多个属性更新时回调函数仅触发一次,避免性能浪费。
- immediate:在初始化时直接调用回调函数,可以通过在 created 阶段手动调用回调函 数实现相同的效果。
- 适用场景
computed:需要处理复杂逻辑的模板表达式。
watch:需要执行异步或开销较大的操作。
从表现上看,computed 会创建新的属性,而 watch 可以通过将属性设置在 data 中,再监 听依赖属性变化,调用 handler 方法更新属性的方式达到同样的效果。因此不难得出 computed 的使用场景可以被 watch 覆盖这一结论。但在具体的使用上还是优先考虑 computed,因为相同场景下 watch 所需的代码量和性能开销一般来说会比 computed 大, 具体可以参照 computed vs watched。在 computed 无法满足需求的情况下再考虑使用 watch,也可以有效避免watch 滥用,提升性能。