Vue响应式原理的实现-面试必问
Vue2的数据响应式原理
1、什么是defineProperty? defineProperty是设置对象属性,利用属性里的set和get实现了响应式双向绑定; 语法:Object.defineProperty(要设置的对象,要修改的对象的属性,属性描述) 属性描述:
1、configurable - 表示此属性能否被delete,默认false;
2、enumerable - 表示此属性能否被枚举,默认为false;
3、value - 设置此属性对应的值,默认为undefined;
4、writable - 设置value属性能否被修改值,为true时方可被改变,默认为false;
5、get - 给属性提供getter方法,默认为undefined,访问该属性时,该方法会被执行,默认参数为this对象;
6、set - 给属性提供setter方法,默认为undefined,属性值修改时,会执行该方法,唯一参数为新的值; 获取对象所拥有属性的方法:Object.getOwnPropertyDescriptor(对象,对象某一属性) var obj = {
a: 1,
b: 2
}
var value = obj.b;
Object.defineProperty(obj,'b',{
configurable: false,//不能被deleted
enumerable: false,//不能被删除
writable: true,//可以被修改值
value: '22',//设置b为22
set: function(newValue){
console.log('新的value为'+ newValue);
value = newValue;
},
get: function(){
console.log('你设置了value');
return value;
},
})
console.log(Object.getOwnPropertyDescriptor(obj,'a')); //{value: 1, writable: true, enumerable: true, configurable: true} 2、实现双向绑定:
function vue(){
this.$data = {a: 1};//设置data
this.el = document.getElementById('app'); //设置根节点
this.dom = '';//虚拟dom
} //往原型上绑定监听方法
vue.prototype.observe = function(obj){
var self = this;
var value;
for(var key in obj){
value = obj[key];
if(typeof value === 'object'){ //如果是对象 需要递归循环执行此方法
this.observe(value);
}else{
Object.defineProperty(this.$data,key,{
get: function(){
//依赖收集 这里略过 代替源码里的 depend
return value;
},
set: function(newVal){
value = newVal;
self.render();//触发更新 代替源码里的dep.notify
}
})
}
}
} vue.prototype.render = function(){
this.dom = '我是' + this.$data.a;
this.el.innerHTML = this.dom;
} 3、数组监听实现
vue数组的特性: push shift unshift
对象的监听是通过defineProperty,而数组是通过dependArray
Object.create(proto,propertiesObject) //方法创建一个新对象,使用现有的对象来提供新创建的对象的__proto__ var arrPro = Array.prototype;
var arrObj = Object.create(arrPro);
var arr = ['push','pop','shift'];
//装饰者模式
arr.forEach(function(method,index){
arrObj[method]=function(){
var ret = arrPro[method].apply(this,arguments);
dep.notify();
return ret;
}
})
Vue3的数据相应原理
let p = new Proxy(target, handler);
target: 目标对象
handler: 是一个对象,其属性是当执行一个操作时定义代理的行为的函数。
handler中常用的对象方法如下:
receiver => 此Proxy对象本身,也就是this指向;
1. get(target, propKey, receiver)
2. set(target, propKey, value, receiver)
3. has(target, propKey)
4. construct(target, args)
5. apply(target, object, args) 写法如下:
let a = new Proxy({},{
get: function(target,key,receiver){
return Reflect.get(target[key]);
//return target[key];
},
set: function(target, key, value, receiver){
return Reflect.set(target,key,value);
//return target[key] = value;
}
}) Object.defineProperty 存在如下缺点:
1. 监听数组的方法不能触发Object.defineProperty方法中的set操作(如果要监听的到话,需要重新编写数组的方法)。
2. 必须遍历每个对象的每个属性,如果对象嵌套很深的话,需要使用递归调用。 Proxy相比defineProperty 区别:
1、参数不一样,defineProperty第二个参数需要指定操作对象的具体key值,而Proxy指定的是最外层对象,这样就省去了循环;
2、defineProperty操作的是对象本身,改变了对象本身,而Proxy是改变对象代理,返回新对象;
3、defineProperty必须在最外层定义一个全局变量value,通过get将其return出去,而Proxy将返回值放到了参数中,直接return target[key];
Diff算法和virtual dom
注:virtual dom是一个虚拟层,并不正式存在
diff算法是直接去比对元素,元素里包含props 和 children, 一直进行到根节点,template中所有的节点都会有个diff,挂载了所绑定的一系列属性,通过遍历循环监听到变化,在vue3中作者改变了这种方式,在发布会中讲到对比vue2速度提升了6倍,
vue2的虚拟dom,会遍历所有子节点,全部用diff算法比对一次,遍历速度由dom数量决定;
vue3中会找出变化的项,只更新改变项;
Vue响应式原理的实现-面试必问的更多相关文章
- 深度解析 Vue 响应式原理
深度解析 Vue 响应式原理 该文章内容节选自团队的开源项目 InterviewMap.项目目前内容包含了 JS.网络.浏览器相关.性能优化.安全.框架.Git.数据结构.算法等内容,无论是基础还是进 ...
- vue响应式原理,去掉优化,只看核心
Vue响应式原理 作为写业务的码农,几乎不必知道原理.但是当你去找工作的时候,可是需要造原子弹的,什么都得知道一些才行.所以找工作之前可以先复习下,只要是关于vue的,必定会问响应式原理. 核心: / ...
- Vue源码--解读vue响应式原理
原文链接:https://geniuspeng.github.io/2018/01/05/vue-reactivity/ Vue的官方说明里有深入响应式原理这一节.在此官方也提到过: 当你把一个普通的 ...
- 详解Vue响应式原理
摘要: 搞懂Vue响应式原理! 作者:浪里行舟 原文:深入浅出Vue响应式原理 Fundebug经授权转载,版权归原作者所有. 前言 Vue 最独特的特性之一,是其非侵入性的响应式系统.数据模型仅仅是 ...
- 深入Vue响应式原理
深入Vue.js响应式原理 一.创建一个Vue应用 new Vue({ data() { return { name: 'yjh', }; }, router, store, render: h =& ...
- vue响应式原理解析
# Vue响应式原理解析 首先定义了四个核心的js文件 - 1. observer.js 观察者函数,用来设置data的get和set函数,并且把watcher存放在dep中 - 2. watcher ...
- 浅析Vue响应式原理(三)
Vue响应式原理之defineReactive defineReactive 不论如何,最终响应式数据都要通过defineReactive来实现,实际要借助ES5新增的Object.definePro ...
- 深入解析vue响应式原理
摘要:本文主要通过结合vue官方文档及源码,对vue响应式原理进行深入分析. 1.定义 作为vue最独特的特性,响应式可以说是vue的灵魂了,表面上看就是数据发生变化后,对应的界面会重新渲染,那么响应 ...
- 浅谈vue响应式原理及发布订阅模式和观察者模式
一.Vue响应式原理 首先要了解几个概念: 数据响应式:数据模型仅仅是普通的Javascript对象,而我们修改数据时,视图会进行更新,避免了繁琐的DOM操作,提高开发效率. 双向绑定:数据改变,视图 ...
随机推荐
- JS对象—对象总结(创建、属性、方法)
1.创建对象Object 1.1 字面量的方式创建 1.2 new Object() 1.3 构造函数创建 1.4 工厂模式 1.5 Object.create() ES5新增方法 Object. ...
- poj2377Bad Cowtractors (最小生成树变形之——最大生成树)
题目链接:http://poj.org/problem?id=2377 Description Bessie has been hired to build a cheap internet netw ...
- 线程局部存储tls的使用
线程局部存储(Thread Local Storage,TLS)主要用于在多线程中,存储和维护一些线程相关的数据,存储的数据会被关联到当前线程中去,并不需要锁来维护.. 因此也没有多线程间资源竞争问题 ...
- 基于Select模型的Windows TCP服务端和客户端程序示例
最近跟着刘远东老师的<C++百万并发网络通信引擎架构与实现(服务端.客户端.跨平台)>,Bilibili视频地址为C++百万并发网络通信引擎架构与实现(服务端.客户端.跨平台),重新复习下 ...
- CentOS tcpdump的使用实例
tcpdump是一个用于截取网络分组,并输出分组内容的工具.tcpdump凭借强大的功能和灵活的截取策略,使其成为类UNIX系统下用于网络分析和问题排查的首选工具. 选项: -A 以ASCII格式打印 ...
- KMP字符串匹配学习
KMP字符串匹配学习 牛逼啊 SYC大佬的博客
- 4、numpy——创建数组
1.普通创建——np.array() 创建数组最简单的方法就是使用array函数.它接收一切序列型的对象(包括其他数组),然后产生一个新的含有传入数据的Numpy数组. import numpy as ...
- 2019牛客暑期多校训练营(第一场) - H - XOR - 线性基
https://ac.nowcoder.com/acm/contest/881/H 题意: 给定n个整数,求其中异或和为 \(0\) 的子集的大小的和. 题解思路: 首先转化为每个可以通过异或表示 \ ...
- Codeforces 1156D 0-1-Tree ( 并查集 || 树形DP )
<题目链接> 题目大意: 给定一颗无向树,树的边权只要0/1两种情况,现在问你这棵树上存在多少对有序对<u,v>,满足u-->v的路径上,如果出现边权为1的边之后,就不能 ...
- 一些常见的js校验
今天有时间来总结一下那些常用的js校验: vernull = function(value){//非空校验 if(value.trim(value).length == 0){ return fals ...