最近看了 Vue 源码和源码分析类的文章,感受明白了不少,可是仔细想一想却说不出个因此然。vue
因此打算把本身掌握的知识,试着组织成本身的语言表达出来dom
不打算平铺直叙的写清楚 vue 源码的来龙去脉和所有细节,而是以自问自答的形式,回答我本身以前的疑惑,异步
若是有错误的地方,欢迎指正哈~函数
Vue 实现响应式的核心 API 是 ES5 的 Object.defineProperty(obj,key,descriptor),Vue 的「响应式」和「依赖收集」都依靠这个 API源码分析
它接受 3 个参数,分别是 obj / key / 描述符,返回的是一个包装后的对象性能
它的做用就是,用这个 API 包装事后的对象能够拥有 getter 和 setter 函数。this
getter 会在对象的这个 key 被获取时触发,setter 会在这个对象的 key 被修改时触发。code
一个 Vue 项目的开始, 一般是从 Vue 构造函数的实例化开始的。对象
new Vue()的时候会执行一个_init()方法,会初始化属性,好比 props/event/生命周期钩子,也包括 data 对象的初始化。递归
Vue 在初始化时,将 data 对象上的全部 key,都包装成拥有 getter 和 setter 的属性。
function cb() { console.log("更新视图"); } function defineReactve(obj, key, val) { Object.defineProperty(obj, key, { enumerable: true, configurable: true, get: () => { console.log("触发了getter"); return val; }, set: newVal => { console.log("触发了setter"); if (newVal === val) return; val = newVal; cb() } }); } function observe(data) { function walk(data) { Object.keys(data).forEach(key => { if (typeof data[key] === "object") { walk(data[key]); } else { defineReactve(data, key, data[key]); } }); } walk(data); } class Vue { constructor(options) { this._data = options.data; observe(this._data); } } var vm = new Vue({ data: { msg: "test", person: { name: "ziwei", age: 18 } } }); vm._data.person.name = 'hello' // 触发setter和cb函数,从而视图更新
function defineReactve( obj, key, val ) { const dep = new Dep() Object.defineProperty( obj, key, { enumerable: true, configurable: true, get: () => { console.log( "触发了getter" ); dep.addSub(Dep.target) return val; }, set: newVal => { console.log( "触发了setter" ); if ( newVal === val ) return; val = newVal; dep.notify() // 通知队列的wather去update视图 } } ); } function observe( data ) { function walk( data ) { Object.keys( data ).forEach( key => { if ( typeof data[ key ] === "object" ) { walk( data[ key ] ); } else { defineReactve( data, key, data[ key ] ); } } ); } walk( data ); } class Dep{ constructor(){ this.subs = [] } addSub(){ this.subs.push(Dep.target) } notify(){ this.subs.forEach(sub => { sub.update() }) } } Dep.target = null class Watcher{ constructor(){ Dep.target = this } update(){ console.log('update更新视图啦~') } } class Vue { constructor( options ) { this._data = options.data; observe( this._data ); new Watcher() // 模拟页面渲染,触发getter,依赖收集的效果 this._data.person.name } } var vm = new Vue( { data: { msg: "test", person: { name: "ziwei", age: 18 } } } ); vm._data.person.name = 'hello'
这样就是 Vue 响应式的一个基本原理,不过我描述的过程当中,也省略了不少环节,好比
Vue 是如何实现给 data 对象上的属性都拥有 getter 和 setter 的
经过循环data对象,给对象的每个key,用Object.defineProperty包装 遍历时,若是发现data[key]也是对象的话,须要用递归
为何要进行「依赖收集」?
举2个场景的栗子🌰
但问题在于,你并不知道谁依赖我?我应该更新哪几个地方
如何避免重复「收集依赖」
在往Dep中放wather对象是,实际上wather的update方法时,会把其放入queue队列中,会经过watch.id判断是否重复了,重复的wather就不会被推入队列
watcher 调用 update,也并非直接更新视图。实现上中间还有 patch 的过程以及使用队列来异步更新的策略。
异步更新的策略,相似于setTimeout(fn,0) ,目的就是为了不频繁的更新dom,让页面的渲染的性能更好。