茗空间

聊聊Vue的数据绑定

黑幕
没图片太单调,用啥图呢,思考片刻选择了这幅图

数据绑定的本质

实现数据绑定的本质就是Setter+change事件,前者Setter用于在数据模型变化时更新UI,后者change事件,用于在UI变化时更新数据模型,来看个大某:

Demo1

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
// 数据模型
var data = {
text: 'Hello World'
};
// UI元素
var input = document.getElementById('input'),
label = document.getElementById('lbl');
// Setter
Object.defineProperty(data, 'text', {
set: function(val) {
label.innerText = val;
input.value = val; // This won't trigger 'change' event.
}
});
// change事件
input.addEventListener('change', function() {
data.text = input.value;
});

从上面的代码可以看出,在data.text的Setter方法中更改了UI元素label和input的值,这样在data.text被赋值时,就会同步更新UI元素;同样在change事件中,input的值改变会同步更新data.text

Vue数据绑定

Demo2

这里只聊聊如何实现当数据变化时更新页面,至于当UI内容变化如何更新数据,其实和上面的例子是一样的。那你肯定会问,难道数据变化时更新页面难道和上面的例子不一样吗?当然不一样,继续看(你不要凑字数好吗?好的)。

我先来说说原理,接下来再上代码。尤大在实现数据变化更新UI时用到了动态收集依赖Dep,下面让尤大来给我们解释解释:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
我:什么是依赖Dep?
尤大:每个数据data的属性property都对应一个依赖Dep,Dep最终会与Watcher相互关联,当property变化时,Dep就会通知watcher执行回调函数。
我:那当数据变化时,触发的所有操作都在watcher的回调函数中吗?
尤大:是的,与传统的data与watcher硬绑定不同,vue在data与watcher之间插入了Dep层,是为了解耦data与watcher,可以随心所欲的修改data与watcher的关联,这种实现方式可以成为依赖的动态收集。
我:哦,原来Dep是用来解耦的,那Dep和Watcher是如何关联的?
尤大:上面我已经提到了data的每个属性property都对应一个Dep,当property被访问时(也就是调用它的getter方法),它的Dep就会添加到当前的watcher。
我:当前的watcher是什么意思?
尤大:表急,耐心听。每个watcher都有一个expOrFn,当expOrFn执行前会把该watcher保存在Dep.target上,这个Dep.target就是当前的watcher。当expOrFn执行时如果访问了某个data的属性property,这个property的Dep就会与Dep.target关联起来。
我:Dep.target是什么?
尤大:Dep.target是Dep的静态属性。
我:哦,好的,全明白了。
尤大:全明白了吗?你应该忘了问Observer了吧。
我:哦,是的,确实忘了,那Observer是什么?
尤大:Observer的作用就是为data的property设置setter和getter,并且定义Dep。

以下是精简后的代码,我把其他细节全部去掉了,只留下真正的核心代码。

Dep(依赖)

dep是用来连接data与watcher的桥梁,每个data的属性都对应一个dep。

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
let uid = 0;
export default class Dep {
// 存放当前的watcher
static target;
constructor() {
// 保存的是Watcher实例
this.subs = [];
// 唯一标示
this.id = uid++;
}
// 添加watcher
addSub(sub) {
this.subs.push(sub);
}
// 移除watcher
removeSub(sub) {
const idx = this.subs.indexOf(sub);
this.subs.splice(idx, 1);
}
depend() {
if (Dep.target) {
Dep.target.addDep(this);
}
}
// 执行所有watcher的run方法
notify() {
const subs = this.subs.slice();
for (let i = 0, l = subs.length; i < l; i++) {
subs[i].run();
}
}
}
// 保存当前watcher
Dep.target = null
export function pushTarget(_target) {
Dep.target = _target;
}
export function popTarget() {
Dep.target = null;

Watcher

当expOrFn执行时,收集dep,保存data属性变化时调的callback。

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
export default class Watcher {
constructor(expOrFn, cb) {
// 当前依赖
this.deps = [];
this.depIds = new Set();
// 新关联的依赖
this.newDeps = [];
this.newDepIds = new Set();
// 回调函数
this.cb = cb;
this.getter = expOrFn; // 只考虑expOrFn是函数的情况
this.value = this.get(); // 这里value其实没用到
}
/**
* 重新收集依赖
*/
get() {
// 将当前watcher放到Dep.target
pushTarget(this);
const value = this.getter(); // 这里value其实没用到
popTarget();
this.cleanupDeps();
return value;
}
/**
* 添加一个依赖
*/
addDep(dep) {
const id = dep.id;
if (!this.newDepIds.has(id)) {
this.newDepIds.add(id);
this.newDeps.push(dep);
if (!this.depIds.has(id)) {
dep.addSub(this);
}
}
}
/**
* 整理新依赖和旧依赖
*/
cleanupDeps() {
let i = this.deps.length
while (i--) {
const dep = this.deps[i]
if (!this.newDepIds.has(dep.id)) {
dep.removeSub(this)
}
}
let tmp = this.depIds;
this.depIds = this.newDepIds;
this.newDepIds = tmp;
this.newDepIds.clear();
tmp = this.deps;
this.deps = this.newDeps;
this.newDeps = tmp;
this.newDeps.length = 0;
}
/**
* 当依赖有变化时就会执行这里
*/
run() {
const value = this.get(); // value其实没用到
this.cb();
}
/**
* 将依赖添加到Dep.target
*/
depend() {
let i = this.deps.length;
while (i--) {
this.deps[i].depend();
}
}
}

Observer

observer使得data是可观察的,为data的每个属性都添加了dep,这里你会见到期盼已久的用Object.defineProperty定义的setter方法,在setter方法里通过dep.notify最终通知watcher执行回调方法。

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
export default class Observer {
constructor(value) {
this.value = value;
this.walk(value);
}
//遍历obj的所有属性,设置其setter和getter
walk(obj) {
const keys = Object.keys(obj)
for (let i = 0; i < keys.length; i++) {
defineReactive(obj, keys[i], obj[keys[i]])
}
}
}
export function defineReactive(obj, key, val) {
// obj中每个key都对应一个dep
const dep = new Dep();
// 将key对应的值保存在__key里,比如obj['name'] = 'glm',则用obj['__name']来保存'glm'。
obj[`__${key}`] = val;
// 定制getter和setter
Object.defineProperty(obj, key, {
enumerable: true,
configurable: true,
get: function reactiveGetter() {
// 当Watcher调其自身的get(️注意不是getter)方法时,将Dep.target赋值为该Watcher。
if (Dep.target) {
// dep.depend作用是让watcher(即Dep.target)与此dep互相引用
// 伪代码是酱的:
// dep.depend() {
// target.addDep(dep) {
// dep.addSub(target);
// }
// }
dep.depend();
}
return this[`__${key}`];
},
set: function reactiveSetter(newVal) {
this.__value = newVal;
// data的属性变了,dep通知watcher该执行回调函数了
dep.notify()
}
})
}

到现在把关键的三个类看完了,来看看如何使用它们:

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
let obj = {
name: 'glm',
sex: 'male'
};
let observer = new Observer(obj);
const watcher = new Watcher(() => {
console.log('-读取过的属性会与watcher建立联系-');
console.log(obj.name);
console.log('-----------------------------');
}, () => {
console.log('Watcher回调函数');
});
console.log('设置name');
obj.name = 'zwr';
console.log('设置sex');
obj.sex = 'female';
// 打印
-读取过的属性会与watcher建立联系-
glm
-----------------------------
设置name
-读取过的属性会与watcher建立联系-
glm
-----------------------------
Watcher回调函数
设置sex

从上面的例子可以看出,在设置name时,触发了watcher的回调,而在设置sex时,没有触发。

完整代码

总结

到现在已经把相关的细节说完了,虽然我的阐述不多,但是已经加上了必要的注释。vue的数据绑定看似逻辑复杂,使用了动态收集依赖的思想,它的好处就是,dep与watcher的关系不是一成不变的,在更新页面的时候,只有被访问属性的dep才会与当前watcher建立联系,只有这个属性才能出发watcher的回调函数。如有有问题欢迎留言。