震惊!!! 2019年10月5日,尤小右公开了 Vue 3.0 的源代码。源码地址:vue-next,此次更新的主要内容除了自行查看源码还可以在知乎上进行了解尤小右 3.0 RFC,在这两篇的基础上,接下来我将为大家展示最近学习到3.0的内容解读
了解3.0的进步,我们得先了解2.0的响应式原理,如果已经知道其优势劣势的大佬自行跳过~~
看过官方文档的同学都知道Vue 响应式系统的解释: 当你把一个普通的 JavaScript 对象传入 Vue 实例作为
data
选项,Vue 将遍历此对象所有的属性,并使用Object.defineProperty
把这些属性全部转为getter/setter
。Object.defineProperty
是 ES5 中一个无法 shim 的特性,这也就是 Vue 不支持 IE8 以及更低版本浏览器的原因
下面我们将大概先实现vue2.0响应
原理:使用 Object.defineProperty 可以重新定义属性,并且给属性增加 getter 和setter;
// 我们先创建一个对象,然后通过某个方法去监听这个对象,当对象的值改变时,触发操作
let defalutName = ''
let data = {name:''}
// observer监听函数
observer(data)
console.log(data.name);
// expected output: Magic Eno
// 给data里面的name赋值 = "Eno"
data.name = 'Eno'
console.log(data.name); // expected output: Eno
console.log(defalutName); // expected output: Eno
observer的效果要求很简单,就是监听data对象,当data里面的属性值改变时,监听到其改变;
下面实现一个简陋的双向数据绑定,即data的name改变时,defaultName也要改变,实现双向数据绑定,即defalutName与data对象的name双向绑定了
let defalutName = ''
let data = {name:''};
function observer (data) {
//Object.defineProperty直接在对象上定义新属性,或修改对象上的现有属性,然后返回对象。
//不了解的请转MDN文档 https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty
Object.defineProperty(data, 'name', {
get(){
return defalutName
},
set(newValue){
defalutName = newValue
}
});
}
//
observer(data)
console.log(data.name);
// expected output: ''
// 给data里面的name赋值 = "Eno"
data.name = 'Eno'
console.log(data.name); // expected output: Eno
console.log(defalutName); // expected output: Eno
上面我们的observer对象并没有对data的所有值进行监听,接下来我们完善oberver函数如下:
function observer(data){
// 判断是否为对象 如果不是则直接返回,Object.defineProperty是对象上的属性
if(typeof data !== 'object' || data == null){
return data;
}
for(let key in data){
defineReactive(data,key,data[key]);
}
}
function defineReactive(data,key,value){
//Object.defineProperty直接在对象上定义新属性,或修改对象上的现有属性,然后返回对象。
//不了解的请转MDN文档 https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty
Object.defineProperty(data, key, {
get(){
console.log('获取了值') // 在此做依赖收集的操作
return value
},
set(newValue){
if(newValue !== value){
console.log('设置了值')
value = newValue
}
}
});
}
let data = {name:'',age:18};
observer(data)
// 给data里面的name赋值 = "Eno"
data.name = 'Eno'
console.log(data.name);
// 设置了值
// 获取了值
// Eno
data.age = 12
console.log(data.age);
// 设置了值
// 获取了值
// 12
补充:什么是依赖收集? 我们都知道,当一个可观测对象的属性被读写时,会触发它的getter/setter方法。如果我们可以在可观测对象的getter/setter里面,执行监听器里面的update()方法;不就能够让对象主动发出通知了吗?
// ...
let data = {name:'',age:18};
observer(data)
// 给data里面的name赋值 = "Eno"
// data.name = 'Eno'
// console.log(data.name);
// data.age = 12
// console.log(data.age);
data.gender = '男'
由图可知,并没有触发set和get,这个因为,在我们对data进行监测的时候是没有gender这个属性值的,因此我们如果想要对新增的属性进行监听的话,需要在赋值后再进行一次监听,即vm.$set的效果;我们可以创建一个reactiveSet函数如下:
function reactiveSet (data,key,value) {
data[key] = value
observer(data)
}
let data = {name:'',age:18};
observer(data)
// 给data里面的name赋值 = "Eno"
// data.name = 'Eno'
// console.log(data.name);
// data.age = 12
// console.log(data.age);
// 通过reactiveSet添加属性
reactiveSet(data,'gender','男')
console.log(data.gender)
此时是可以响应的,不过vue并不是这样做的,下面可以看vue的源码
vuejs in github
里面是这样判断的,如果这个key目前没有存在于对象中,那么会进行赋值并监听。但是这里省略了ob的判断;
补充:
ob是什么呢?
vue初始化的数据(如data中的数据)在页面初始化的时候都会被监听,而被监听的属性都会被绑定__ob__属性,下图就是判断这个数据有没有被监听的。如果这个数据没有被监听,那么就默认你不想监听这个数据,所以直接赋值并返回
目前,我们是对data一个简单对象进行监听,思考一下假如是多层对象该如何调整及修改observer呢?
// ...
function defineReactive(data,key,value){
observer(value); // 递归 继续对当前value进行拦截
//Object.defineProperty直接在对象上定义新属性,或修改对象上的现有属性,然后返回对象。
//不了解的请转MDN文档 https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty
Object.defineProperty(data, key, {
get(){
console.log('获取了值') // 在此做依赖收集的操作
return value
},
set(newValue){
if(newValue !== value){
// 对于新增的值也需要监听
observer(newValue)
console.log('设置了值')
value = newValue
}
}
});
}
// ...
2019年10月21日更新
假如data里面的对象里面有数组,那么需要对数组进行拦截,如果数组里面是多维数组,还需和5.嵌套对象的做法一致,还需要进行递归监听,observer修改如下:
function observer(data){
// 判断是否为对象 如果不是则直接返回,Object.defineProperty是对象上的属性
if(typeof data !== 'object' || data == null){
return data;
}
if(Array.isArray(data)){ // 如果是数组,则对数据进行遍历并对其value进行递归监听
for(let i = 0; i< data.length ;i++){
observer(data[i]);
}
} else {
for(let key in data){
defineReactive(data,key,data[key]);
}
}
}
// ...
// ...
let data = {name:'',age:18};
observer(data)
// 给data里面的name赋值 = "Eno"
// data.name = 'Eno'
// console.log(data.name);
// data.age = 12
// console.log(data.age);
// reactiveSet(data,'gender','男')
// console.log(data.gender)
reactiveSet(data,'attr',[1,2,3,4,5,100])
console.log(data.attr)
从上图可以看出,在reactiveSet的情况下,即使给data设置了不存在的数组,也能够得到监听,接下来尝试对数组进行修改测试;
// ...
let data = {name:'',attr:[1,2,3,4,5,100]};
observer(data)
// 给data里面的name赋值 = "Eno"
// data.name = 'Eno'
// console.log(data.name);
// data.age = 12
// console.log(data.age);
// reactiveSet(data,'gender','男')
// console.log(data.gender)
// console.log(data.attr)
data.attr.push(10000);
data.attr.splice(0,1);
究其原因:由于数组的方法在对数组的增删查改过程中,vue并没其操作更新视图的操作,故此时是不能响应式的,因此如果需要对此类方法的调用时,通过视图更新,则需要对数组方法
重写
,查看vue/array.js源码可知:
const arrayProto = Array.prototype
const arrayMethods = Object.create(arrayProto)
const methodsToPatch = [
'push',
'pop',
'shift',
'unshift',
'splice',
'sort',
'reverse'
]
methodsToPatch.forEach(method=>{
arrayMethods[method] = function(){
// 函数劫持 把函数进行重写
// 而内部实际上继续调用原来的方法但在这里我们可以去调用更新视图的方法
console.log('数组 更新啦...')
arrayProto[method].call(this,...arguments)
}
});
function observer(data){
// 判断是否为对象 如果不是则直接返回,Object.defineProperty是对象上的属性
if(typeof data !== 'object' || data == null){
return data;
}
if(Array.isArray(data)){ // 如果是数组,则对数据进行遍历并对其value进行递归监听
// 在这里对数组方法进行重写 即函数劫持
Object.setPrototypeOf(data, arrayMethods);
for(let i = 0; i< data.length ;i++){
observer(data[i]);
}
} else {
for(let key in data){
defineReactive(data,key,data[key]);
}
}
}
const arrayProto = Array.prototype
const arrayMethods = Object.create(arrayProto)
const methodsToPatch = [
'push',
'pop',
'shift',
'unshift',
'splice',
'sort',
'reverse'
]
methodsToPatch.forEach(method=>{
arrayMethods[method] = function(){
//函数劫持 把函数进行重写
// 而内部实际上继续调用原来的方法但在这里我们可以去调用更新视图的方法
console.log('数组 更新啦...')
arrayProto[method].call(this,...arguments)
}
});
function observer(data){
// 判断是否为对象 如果不是则直接返回,Object.defineProperty是对象上的属性
if(typeof data !== 'object' || data == null){
return data;
}
if(Array.isArray(data)){ // 如果是数组,则对数据进行遍历并对其value进行递归监听
// 在这里对数组方法进行重写 即函数劫持
Object.setPrototypeOf(data, arrayMethods);
for(let i = 0; i< data.length ;i++){
observer(data[i]);
}
} else {
for(let key in data){
defineReactive(data,key,data[key]);
}
}
}
function defineReactive(data,key,value){
observer(value); // 递归 继续对当前value进行拦截
//Object.defineProperty直接在对象上定义新属性,或修改对象上的现有属性,然后返回对象。
//不了解的请转MDN文档 https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty
Object.defineProperty(data, key, {
get(){
console.log('获取了值') // 在此做依赖收集的操作
return value
},
set(newValue){
if(newValue !== value){
// 对于新增的值也需要监听
observer(newValue)
console.log('设置了值')
value = newValue
}
}
});
}
function reactiveSet (data,key,value) {
data[key] = value
observer(data)
}
let data = {name:'',attr:[1,2,3,4,5,100]};
observer(data)
// 给data里面的name赋值 = "Eno"
// data.name = 'Eno'
// console.log(data.name);
// data.age = 12
// console.log(data.age);
// reactiveSet(data,'gender','男')
// console.log(data.gender)
// console.log(data.attr)
data.attr.push(10000);
data.attr.splice(0,1);
看到视图真的更新了,不得不佩服尤大大真的厉害,听说2020年第一季度就要出vue3.0了,接下来要写篇vue3.0的初步学习文章,希望各位看官支持,不要忘记点赞喔;
文章源码:github wLove-c
总结: 能力有限,暂时先写这么多,接下来有时间会写一篇vue-next的源码实践和理解, 希望各位看官大人不要忘记点赞哈,写的不好的地方欢迎指正;
Document