vue响应式原理简述(vue的响应式原理)

技术vue2和vue3数据响应式原理分析及如何实现今天就跟大家聊聊有关vue2和vue3数据响应式原理分析及如何实现,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。

今天给大家讲讲vue2和vue3数据响应的原理分析和实现,可能很多人都不太懂。为了让大家更好的了解,边肖为大家总结了以下内容,希望大家能从这篇文章中有所收获。

00-1010视图和数据会自动更新,数据更新时视图也会自动更新。

跟踪数据的变化,在读取数据或设置数据时做一些劫持操作。

Vue2使用定义属性

Vue3已更改为代理

数据响应式

使用defineProperty

varobj={ } varage

Object.defineProperty(obj,' age ',{ 0

get : function(){ 0

consoel.log('getage . ')

返回},

set : function(val){ 0

console.log('setage . ')

Age=val }}) obj.age=100//setage.console.log (obj.age)//getage.对象obj在获取age属性时会调用数据劫持的get方法。

当年龄属性被赋值时,调用set方法。

那么如何使用Object.defineProperty来实现数据响应呢?

函数定义活动(数据){ 0

if(!data | | object . prototype . tostring . call(数据)!=='[objectobjectobject]')

返回;

for(letkeyindata){ 0

let val=data[key];

Object.defineProperty(数据,键,{ 0

Enumerable:true,//enumerable

可配置:真实,//可配置

get : function(){ 0

轨道(数据、键);

returnval

},

set : function(){ 0

触发器(val,key);

},

});

if(type of val==' object '){ 0

定义活动

(val);
    }
  }}function trigger(val, key) {
  console.log("sue set", val, key);}function track(val, key) {
  console.log("sue set", val, key);}const data = {
  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个,分别push pop shift unshift splice sort reverse
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对象用于创建一个对象的代理, 从而实现基本操作的拦截和定义(如属性查找、赋值、枚举、函数掉用等)

const p = new Proxy(target, handler)
  • target

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

  • handler

  • 一个通常以函数作为属性的对象,各属性中的函数分别定义了在执行各种操作时代理 p 的行为。
    reflect是一个内置对象, 他提供拦截javascript操作的方法, 这些方法和Proxy handlers相同

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

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

如何实现劫持

const dinner = {
  meal:'111'}const handler = {
  get(target, prop) {
    console.log('get...', prop)
    return Reflect.get(...arguments)
  },
  set(target, key, value) {
    console.log('get...', prop)
    console.log('set',key,value)
    return Reflect.set(...arguments)
  }}const proxy = new Proxy(dinner, handler)console.log(proxy.meal)console.log(proxy.meal)

代码中dinner 对象代理到handler上
defineProperty区别
defineProperty的属性需要遍历才能监管所有属性

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

用proxy实现一个模拟响应式

function reactive(obj) {
  const handler = {
    get(target, prop, receiver) {
      track(target, prop);
      const value =  Reflect.get(...arguments);
      if(typeof value === 'Object') {
        reactive(value)
      }else {
        return value      }
    },
    set(target,key, value, receiver) {
      trigger(target,key, value);
      return Reflect.set(...arguments);
    },
  };
  return new Proxy(obj,handler)}function track(data, key) {
  console.log("sue set", data, key);}function trigger(data, key,value) {
  console.log("sue set", key,':',value);}const dinner = {
  name:'haochi1'}const proxy  =reactive(dinner)proxy.name
proxy.list = []proxy.list.push(1)

执行后自动打印

vue2和vue3数据响应式原理分析及如何实现

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

赋值也需要先get

简单总结:

  1. vue2 (浅响应式)

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

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

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

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

  1. vue3 (深度响应式) :

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

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

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

两者的不同

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

看完上述内容,你们对vue2和vue3数据响应式原理分析及如何实现有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注行业资讯频道,感谢大家的支持。

内容来源网络,如有侵权,联系删除,本文地址:https://www.230890.com/zhan/155754.html

(0)

相关推荐

  • Java远程过程调用方法是什么

    技术Java远程过程调用方法是什么本篇内容介绍了“Java远程过程调用方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有

    攻略 2021年12月8日
  • 农村医保网上如何缴费,农村医疗保险网上怎么交

    技术农村医保网上如何缴费,农村医疗保险网上怎么交人登录市地税局网站后农村医保网上如何缴费,可通过“网上办税(费)厅”,进入“自助缴费”模块,采用“银税转账”或“网上银行”两种划款方式完成缴费。1.若使用“银税转账”方式,

    生活 2021年10月30日
  • 软件设计-策略模式

    技术软件设计-策略模式 软件设计-策略模式旅行方式的选择旅游的出行方式有乘坐飞机旅行、乘火车旅行和自行车游,不同的旅游方式有不同的实现过程,客户可以根据自己的需要选择一种合适的旅行方式。类图Javapa

    礼包 2021年12月14日
  • utf8转ansi编码在线转换(ansi转utf8在线)

    技术如何解决KEIL编译器的UTF-8和ANSI的转换问题这期内容当中小编将会给大家带来有关如何解决KEIL编译器的UTF-8和ANSI的转换问题,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有

    攻略 2021年12月23日
  • 如何解决MacBook pro忘记开机密码的问题

    技术如何解决MacBook pro忘记开机密码的问题这篇文章将为大家详细讲解有关如何解决MacBook pro忘记开机密码的问题,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。场景:有一

    攻略 2021年11月17日
  • ios 动态gif制作(ios可以实现动画效果的几种方式)

    技术iOS的GIF动画效果怎么实现本篇文章给大家分享的是有关iOS的GIF动画效果怎么实现,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。GIF在iOS中的使

    攻略 2021年12月18日