vue2&vue3数据响应式原理分析及手动实现(实例详解)

本篇文章给大家带来了vue2&vue3数据响应式原理分析及手动实现的相关知识,数据响应式视图跟数据是自动更新的,数据更新的时候视图是自动的更新的追踪数据的变化,希望对大家有…

本篇文章给大家带来了vue2&vue3数据响应式原理分析及手动实现的相关知识,数据响应式视图跟数据是自动更新的,数据更新的时候视图是自动的更新的追踪数据的变化,希望对大家有帮助。

vue2&vue3数据响应式原理分析及手动实现(实例详解)

数据响应式

视图跟数据是自动更新的,数据更新的时候视图是自动的更新的

追踪数据的变化,在读取数据或者设置数据的时候能够做一些劫持的一些操作

vue2 使用defineProperty

vue3 改用Proxy

使用defineProperty

如何追踪变化

varobj={}varageObject.defineProperty(obj,'age',{get:function(){consoel.log('getage…')returnage},set:function(val){console.log('setage…')age=val}})obj.age=100//setage…console.log(obj.age)//getage…

对象obj在取age属性的时候会调用数据劫持的get方法

在给age属性赋值的时候会调用set方法

那怎么使用Object.defineProperty实现一个数据响应式呢

functiondefineReactive(data){if(!data||Object.prototype.toString.call(data)!=='[objectObject]')return;for(letkeyindata){letval=data[key];Object.defineProperty(data,key,{enumerable:true,//可枚举configurable:true,//可配置get:function(){track(data,key);returnval;},set:function(){trigger(val,key);},});if(typeofval==="object"){defineReactive(val);}}}functiontrigger(val,key){console.log("sueset",val,key);}functiontrack(val,key){console.log("sueset",val,key);}constdata={name:'better',firends:['1','2']}defineReactive(data)console.log(data.name)console.log(data.firends[1])console.log(data.firends[0])console.log(Object.prototype.toString.call(data))

这个函数defineReactve用来对Object.defineProperty进行封装,从函数名可以看出,起作用就是定义一个响应式数据,封装后只需要传递data,key和val就行

每当从data中读取key的时候触发track函数,往data的key中设置数据时,set函数中的trigger函数触发

数组的响应式

我们通过Array原型上的方法来改变数组的内容不会触发getter和setter

整理发现Array原型中可以改变数组自身内容的方法有7个,分别pushpopshiftunshiftsplicesortreverse

vue2 改写了这这7种方法

实现方式:

以Array.propertype为原型创建一个arrayMethods对象,再使用Object.setPropertypeOf(o, arryMethods)将o的__proto__指向arrayMethods

vue2&vue3数据响应式原理分析及手动实现(实例详解)

如何收集依赖

使用

<template><p>{{name}}</p></template>

该模板中使用数据name, 我们要观察数据, 当数据的属性发生变化的时候, 可以通知哪些使用的地方

这就是我们要先收集依赖,即把用到数据name的地方收集起来,然后等数据变化的时候,把之前收集好的依赖循环触发一遍,总结来说就是getter中收集依赖,在setter中触发依赖

使用proxy

Proxy对象用于创建一个对象的代理, 从而实现基本操作的拦截和定义(如属性查找、赋值、枚举、函数掉用等)

constp=newProxy(target,handler)

target

要使用Proxy包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)。

handler

一个通常以函数作为属性的对象,各属性中的函数分别定义了在执行各种操作时代理p的行为。

reflect是一个内置对象, 他提供拦截javascript操作的方法, 这些方法和Proxy handlers相同

Reflect.set将值分配给属性的函数。返回一个Boolean 如果更新成功则返回true

Reflect.get获取对象身上某个属性的值,类似target[name]

如何实现劫持

constdinner={meal:'111'}consthandler={get(target,prop){console.log('get…',prop)returnReflect.get(…arguments)},set(target,key,value){console.log('get…',prop)console.log('set',key,value)returnReflect.set(…arguments)}}constproxy=newProxy(dinner,handler)console.log(proxy.meal)console.log(proxy.meal)

代码中dinner 对象代理到handler上

跟defineProperty区别

defineProperty的属性需要遍历才能监管所有属性

使用proxy可以将对象所有属性进行代理

用proxy实现一个模拟响应式

functionreactive(obj){consthandler={get(target,prop,receiver){track(target,prop);constvalue=Reflect.get(…arguments);if(typeofvalue==='Object'){reactive(value)}else{returnvalue}},set(target,key,value,receiver){trigger(target,key,value);returnReflect.set(…arguments);},};returnnewProxy(obj,handler)}functiontrack(data,key){console.log("sueset",data,key);}functiontrigger(data,key,value){console.log("sueset",key,':',value);}constdinner={name:'haochi1'}constproxy=reactive(dinner)proxy.nameproxy.list=[]proxy.list.push(1)

执行后自动打印

vue2&vue3数据响应式原理分析及手动实现(实例详解)

思考:为啥只在get中使用递归,set不使用呢?

赋值也需要先get

简单总结:

1、vue2 (浅响应式)

遍历data,使用defineProperty拦截所有属性

当用户操作视图,会触发set拦截器

set先改变当前数据, 再通知wartch, 让watch去通知视图更新

视图重绘, 再次从get中获取对应的数据

2、vue3 (深度响应式) :

使用proxy 进行代理;拦截data任意属性的任意操作(13种), 包括属性的读写, 属性的添加, 属性的删除等等

使用Reflect进行反射; 动态对被代理的对象的相应属性进行特定的操作

代理对象(proxy)的反射对象(reflect)必须相互配合才能实现响应式

两者的不同

Proxy能劫持整个对象,而Object.defineProperty只能劫持对象的属性; 前者递归返回属性对应的值的代理即可实现响应式,后者需要深度遍历每个属性,后者对数组的操作很不友好。

产品猿社区致力收录更多优质的商业产品,给服务商以及软件采购客户提供更多优质的软件产品,帮助开发者变现来实现多方共赢;

日常运营的过程中我们难免会遇到各种版权纠纷等问题,如果您在社区内发现有您的产品未经您授权而被用户提供下载或使用,您可按照我们投诉流程处理,点我投诉

本文来自用户发布投稿,不代表产品猿立场 ;若对此文有疑问或内容有严重错误,可联系平台客服反馈;

部分产品是用户投稿,可能本文没有提供官方下下载地址或教程,若您看到的内容没有下载入口,您可以在我们产品园商城搜索看开发者是否有发布商品;若您是开发者,也诚邀您入驻商城平台发布的产品,地址:点我进入

如若转载,请注明出处:https://www.chanpinyuan.cn/39352.html;
(0)
上一篇 2023年3月17日 下午4:19
下一篇 2023年3月17日 下午4:19

相关推荐

发表回复

登录后才能评论
分享本页
返回顶部