前言 我经常使用watch
和watcheffect
,但前几天遇到一个稍微复杂点的功能,就出现了意料之外的情况,感觉对它们了解的还不够。所以,在此详细探索一下它们。
watch vs watchEffect 它们的相同点是都可以监听响应式数据 的变化,并执行回调函数。不同点比较多。 不同点如下:
watch需要显示指定要监听的数据,watchEffect会自动 收集依赖;
watchEffect会立即 执行一次,watch在设置{immediate: true}
时才会立即执行一次;
watch可以获取到旧值 ,watchEffect不可以;
watch可以通过选项设置为仅执行一次 ,watchEffect不可以;
watch可以设置监听对象的层数 ,watchEffect不可以;
官方文档也有指出,watch相对于watchEffect的优点:
与 watchEffect() 相比,watch() 使我们可以:
懒执行副作用;
更加明确是应该由哪个状态触发侦听器重新执行;
可以访问所侦听状态的前一个值和当前值
综上所述,使用watchEffect虽然比较省事,但如果功能复杂很容易出现问题,灵活性也不如watch。比如,在watchEffect内调用一个函数,但这个函数内读取了某个响应式数据,导致这个数据被意外监听。所以,我推荐==任何情况都使用watch ==。
watch详解 第一个参数:侦听源 watch的第一个参数指定要监听哪些响应式数据。这些数据必须是响应式的,但可以包括以下几种:
一个ref
一个响应式对象
一个函数,这个函数返回一个值
由以上类型的值组成的数组
当第一个参数是响应式对象时,默认开启深度监听。 当第一个参数是函数时,该函数会立即执行。 当第一个参数是getter函数时,有以下三种情况:
如果该函数返回响应式对象,回调函数不触发;1 2 3 4 5 6 7 const state = reactive ({ count : 0 });watch ( () => state, (newValue, oldValue ) => { console .log (newValue === oldValue); }, );
如果返回响应式对象的某个属性,回调函数触发;1 2 3 4 5 6 7 const state = reactive ({ count : 0 });watch ( () => state.count , (newValue, oldValue ) => { console .log (newValue === oldValue); }, );
如果返回响应式对象,但开启{deep: true}
,回调函数触发。但,此时新值和旧值相同。1 2 3 4 5 6 7 8 9 10 const state = reactive ({ count : 0 });watch ( () => state, (newValue, oldValue ) => { console .log (newValue === oldValue); }, { deep : true , }, );
第二个参数:回调函数 watch的第二个参数会在监听的数据发生变化时被调用。如果监听的多个数据都发生变化,在一个事件循环周期内回调只执行一次。 该回调函数有3个参数:新值、旧值、一个用来清理副作用 的方法。 当监听数组时,新值和旧值也都是数组,与监听源一一对应。
副作用清理 副作用清理的使用场景:异步未返回时,watch的回调就再次执行。 使用方法:把清理副作用的方法传入到onCleanup
中,如下所示,cancel
并不会马上执行,而是下一次执行watch的回调函数时执行。
1 2 3 4 5 watch (id, async (newId, oldId, onCleanup) => { const { response, cancel } = doAsyncWork (newId) onCleanup (cancel) data.value = await response })
vue3.5+引入了onWatcherCleanup,
1 2 3 4 5 6 7 import { onWatcherCleanup } from 'vue' watch (id, async (newId) => { const { response, cancel } = doAsyncWork (newId) onWatcherCleanup (cancel) data.value = await response })
那么,onCleanup和onWatcherCleanup有什么不同呢?
语义友好,onWatcherCleanup明确表示这个方法与watch的清理操作相关;
onWatcherCleanup更好的和vue3.5+的新特性结合,vue3.5+新增了watch选项、优化了watcher的管理机制;
出现错误时,vue会为onWatcherCleanup返回更详细的错误信息;
==总结:如果你的vue版本是3.5+,请选择onWatcherCleanup。==
第三个参数:配置对象 第三个可选的参数是一个对象,支持以下选项:
immediate: 立即触发回调,第一次调用时旧值是undefined
;
deep: 如果源是对象,进行深度遍历。在3.5+中可以设置为最大遍历深度的数字;
once: 回调函数只执行一次;
flush: 设置回调函数的刷新时机;
onTrack/onTrigger: 调试侦听器的依赖;
前3个选项比较好理解也经常用,接下来了解一下后面2个。
flush flush的值有2个:post
和sync
。 默认情况下,watch的回调函数会在父组件更新之后、所属组件的DOM更新之前 被调用。当你在回调函数中访问所属组件的DOM时,获取到的是更新前的状态 。 当设置flush为post时,在回调函数中获取到的DOM是更新之后的状态。
默认情况下,回调函数会被批量处理,一个事件循环内只执行一次。 当设置flush为sync时,监听的数据变化时就马上触发回调函数。所以要慎重使用,可以监听布尔值,避免监听可能多次同步修改的数据源。
onTrack/onTrigger 注意:这两个方法仅在开发模式下生效。
1 2 3 4 5 6 7 8 watch (source, callback, { onTrack (e ) { debugger }, onTrigger (e ) { debugger } })
onTrack在响应式数据被追踪时调用,onTrigger在响应式数据发生变化时被调用。 可以用这2个方法在开发环境进行调试。
返回值:一个停止监听的函数 watch的返回值可以用来停止、暂停、恢复监听,类型定义如下所示:
1 2 3 4 5 6 interface WatchHandle { (): void pause : () => void resume : () => void stop : () => void }
使用场景:在组件销毁时,或达到某个条件时,停止监听。
watch源码解析 首先解释几个名词,假设监听源是source,监听的回调函数是cb,
traverse : 一个深度优先遍历对象的方法。当source是响应式对象 时,用traverse遍历source,那么响应式对象的每一个属性都执行了读取操作 ,所以当响应式对象任一属性发生变化时都能触发回调函数的执行。
getter : 一个获取source的函数。根据source的类型,对source进行读取操作,使source的变化能被监听。
effect : 副作用函数。 当响应式对象某个属性的值发生变化时,需要重新执行的函数。
onWatcherCleanup : 一个注册清理副作用的函数,接收一个函数作为参数,比如把一个清除定时器的函数传给它。注意这个副作用不是指effect,而是指定时器、异步等,它们会影响下一次cb的执行。
boundCleanup : 是onWatcherCleanup的绑定版本,把effect传给onWatcherCleanup。
cleanup : 执行传给onWatcherCleanup的那些清理副作用的函数。
ReactiveEffect : 一个类,它的实例是effect,它具有的属性和方法如下所示,
active: 布尔值,表示副作用函数是否处于活动状态,source变化时,当active是true,cb才重新执行;
deps: 数组,包含副作用函数的依赖项;
fn: 副作用函数本身;
scheduler: 一个调度器函数,控制副作用函数的执行时机;
scope: 副作用函数的作用域;
run: 一个方法,用于执行副作用函数;
stop: 一个方法,用于停止副作用函数;
pause: 一个方法,用于暂停副作用函数的执行;
resume: 一个方法,用于恢复副作用函数的执行;
onStop: 一个方法,stop被调用时执行;
完整源码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 export function watch (source, cb, options ) { const { immediate, deep, once, scheduler, augmentJob, call } = options; const reactiveGetter = (source ) => { if (deep) return source if (isShallow (source) || deep === false || deep === 0 ) return traverse (source, 1 ) return traverse (source) } let effect; let getter; let cleanup; let boundCleanup; let forceTrigger = false ; let isMultiSource = false ; if (isRef (source)) { getter = () => source.value forceTrigger = isShallow (source) } else if (isReactive (source)) { getter = () => reactiveGetter (source) forceTrigger = true } else if (isArray (source)) { isMultiSource = true forceTrigger = source.some (s => isReactive (s) || isShallow (s)) getter = () => source.map (s => { if (isRef (s)) { return s.value } else if (isReactive (s)) { return reactiveGetter (s) } else if (isFunction (s)) { return call ? call (s, WatchErrorCodes .WATCH_GETTER ) : s () } else { __DEV__ && warnInvalidSource (s) } }) } else if (isFunction (source)) { if (cb) { getter = call ? () => call (source, WatchErrorCodes .WATCH_GETTER ) : source } else { getter = () => { if (cleanup) { pauseTracking () try { cleanup () } finally { resetTracking () } } const currentEffect = activeWatcher activeWatcher = effect try { return call ? call (source, WatchErrorCodes .WATCH_CALLBACK , [boundCleanup]) : source (boundCleanup) } finally { activeWatcher = currentEffect } } } } else { getter = NOOP __DEV__ && warnInvalidSource (source) } if (cb && deep) { const baseGetter = getter const depth = deep === true ? Infinity : deep getter = () => traverse (baseGetter (), depth) } const scope = getCurrentScope () const watchHandle = ( ) => { effect.stop () if (scope && scope.active ) { remove (scope.effects , effect) } } if (once && cb) { const _cb = cb cb = (...args ) => { _cb (...args) watchHandle () } } let oldValue = isMultiSource ? new Array (source.length ).fill (INITIAL_WATCHER_VALUE ) : INITIAL_WATCHER_VALUE const job = (immediateFirstRun ) => { if ( !(effect.flags & EffectFlags .ACTIVE ) || (!effect.dirty && !immediateFirstRun) ) { return } if (cb) { const newValue = effect.run () if ( deep || forceTrigger || (isMultiSource ? newValue.some ((v, i ) => hasChanged (v, oldValue[i])) : hasChanged (newValue, oldValue)) ) { if (cleanup) { cleanup () } const currentWatcher = activeWatcher activeWatcher = effect try { const args = [ newValue, oldValue === INITIAL_WATCHER_VALUE ? undefined : isMultiSource && oldValue[0 ] === INITIAL_WATCHER_VALUE ? [] : oldValue, boundCleanup, ] call ? call (cb, WatchErrorCodes .WATCH_CALLBACK , args) : cb (...args) oldValue = newValue } finally { activeWatcher = currentWatcher } } } else { effect.run () } } if (augmentJob) { augmentJob (job) } effect = new ReactiveEffect (getter) effect.scheduler = scheduler ? () => scheduler (job, false ) : job boundCleanup = fn => onWatcherCleanup (fn, false , effect) cleanup = effect.onStop = () => { const cleanups = cleanupMap.get (effect) if (cleanups) { if (call) { call (cleanups, WatchErrorCodes .WATCH_CLEANUP ) } else { for (const cleanup of cleanups) cleanup () } cleanupMap.delete (effect) } } if (__DEV__) { effect.onTrack = options.onTrack effect.onTrigger = options.onTrigger } if (cb) { if (immediate) { job (true ) } else { oldValue = effect.run () } } else if (scheduler) { scheduler (job.bind (null , true ), true ) } else { effect.run () } watchHandle.pause = effect.pause .bind (effect) watchHandle.resume = effect.resume .bind (effect) watchHandle.stop = watchHandle return watchHandle }