deep原理 vue的watch_【Vue】watch中的deep:true源码实现

当用户指定了watch中的deep属性为true时,如果当时监控的属性是数组类型,会对对象中的每一项进行求值,此时会将当前watcher存入到对应属性的依赖中,这样数组中对象发生变化时也会通知数据更新。内部原理就是递归,耗费性能 。

整体流程:

initWatch 初期化user watcher(1),user watcher在defineReactive的get中订阅属性的变化(2),在defineReactive的set时触发notify(2),notify调用每个订阅了改属性变化的watcher的update(3),监听 watcher 进入update的queueWatcher,在queueWatcher的nextTick中调用flushSchedulerQueue(4),flushSchedulerQueue中调用监听watcher的run(5),在watcher.run中调用this.get,在this.get中判断this.deep是否为true,为true则执行traverse,在traverse中会因为对属性的取值触发2的get方法,并且traverse递归调用,使当前watch监听到对象内部的每一个属性(6),进而调用用户在watch属性上定义的方法。

1、initWatch 初期化user watcher(src\core\instance\state.js)

functioninitWatch (vm: Component, watch: Object) {for (const key inwatch) {

const handler=watch[key]if(Array.isArray(handler)) {for (let i = 0; i < handler.length; i++) {

createWatcher(vm, key, handler[i])//每一项创建一个watcher

}

}else{

createWatcher(vm, key, handler)

}

}

}

2、user watcher在defineReactive的get中订阅属性的变化,在defineReactive的set时触发notify

export functiondefineReactive (

obj: Object,

key: string,

val: any,

customSetter?: ?Function,

shallow?: boolean) {

...

Object.defineProperty(obj, key, {

enumerable:true,

configurable:true,

get:functionreactiveGetter () {

const value= getter ?getter.call(obj) : valif(Dep.target) {

dep.depend()if(childOb) {

childOb.dep.depend()if(Array.isArray(value)) {

dependArray(value)

}

}

}returnvalue

},

set:functionreactiveSetter (newVal) {

...

dep.notify()

}

})

}

3、notify调用每个订阅了改属性变化的watcher的update

notify () {

...for (let i = 0, l = subs.length; i < l; i++) {

subs[i].update()

}

}

4、queueWatcher(src\core\observer\scheduler.js):

export functionqueueWatcher (watcher: Watcher) {

...

nextTick(flushSchedulerQueue)

}

5、调用监听watcher的run

functionflushSchedulerQueue () {

...for (index = 0; index < queue.length; index++) {

...

watcher.run()

...

}

...

}

6、traverse

df99f36061e669ef1c2a2ae1a12cdd5d.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值