结束了之前prepare,现在开始正式进入vue源码探究部分
数据代理先从简单的入手,我们先手动实现一个简易版的数据代理吧。
基础知识vue数据代理: data对象的所有属性的操作(读/写)由vm对象来代理操作
好处: 通过vm对象就可以方便的操作data中的数据
实现:
1). 通过Object.defineProperty(vm, key, {})给vm添加与data对象的属性对应的属性
2). 所有添加的属性都包含get/set方法
3). 在get/set方法中去操作data中对应的属性
举个例子
上面这部分是为了简单说明,我们写的mvvm实现了简单的数据代理,接下来,我们来剖析一下它的内部实现。
先揭一个底,它的核心实现就是之前讲过的Object.defineProperty
function MVVM(options) {
// 将选项对象保存到vm this.$options = options;
// 将data对象保存到vm和datq变量中 var data = this._data = this.$options.data;
//将vm保存在me变量中 var me = this;
// 遍历data中所有属性 Object.keys(data).forEach(function (key) { // 属性名: name // 对指定属性实现代理 me._proxy(key);
});
// 对data进行监视 observe(data, this);
// 创建一个用来编译模板的compile对象 this.$compile = new Compile(options.el || document.body, this)}MVVM.prototype = {
$watch: function (key, cb, options) {
new Watcher(this, key, cb);
},
// 对指定属性实现代理 _proxy: function (key) {
// 保存vm var me = this;
// 给vm添加指定属性名的属性(使用属性描述) Object.defineProperty(me, key, {
configurable: false, // 不能再重新定义 enumerable: true, // 可以枚举 // 当通过vm.name读取属性值时自动调用 get: function proxyGetter() {
// 读取data中对应属性值返回(实现代理读操作) return me._data[key];
},
// 当通过vm.name = 'xxx'时自动调用 set: function proxySetter(newVal) {
// 将最新的值保存到data中对应的属性上(实现代理写操作) me._data[key] = newVal;
}
});
}};Tips:
在源码学习的过程中,掌握到了一个小技巧,读源码的时候,debugger会更加的方便。
image.png
image.png
非常好用,我之前太蠢了,只会第一个resume script execution
最后未完待续...
接下来,还有一个更有趣的东西
下一章继续~



