• 技术文章 >web前端 >Vue.js

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

    长期闲置长期闲置2021-12-22 18:25:28转载98
    本篇文章给大家带来了vue2&vue3数据响应式原理分析及手动实现的相关知识,数据响应式视图跟数据是自动更新的,数据更新的时候视图是自动的更新的追踪数据的变化,希望对大家有帮助。

    数据响应式

    使用defineProperty

    如何追踪变化

    var obj = {}var age 
    Object.defineProperty(obj, 'age', {
        get: function() {
            consoel.log('get age ...')
            return age    },
        set: function(val) {
            console.log('set age ...')
            age = val    }})obj.age =100 //set age ...console.log(obj.age)//get age ...

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

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

    function defineReactive(data) {
      if (!data || Object.prototype.toString.call(data) !== '[object Object]')
        return;
      for (let key in data) {
        let val = data[key];
        Object.defineProperty(data, key, {
          enumerable: true, //可枚举
          configurable: true, //可配置
          get: function() {
            track(data, key);
            return val;
          },
          set: function() {
            trigger(val, key);
          },
        });
        if (typeof val === "object") {
          defineReactive(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

    image.png

    如何收集依赖

    使用

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

    该模板中使用数据 name, 我们要观察数据, 当数据的属性发生变化的时候, 可以通知哪些使用的地方,
    这就是我们要先收集依赖,即把用到数据name的地方收集起来,然后等数据变化的时候,把之前收集好的依赖循环触发一遍,总结来说就是getter中收集依赖,在setter中触发依赖

    使用proxy

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

    const p = new Proxy(target, handler)

    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)

    执行后自动打印

    impicture_20210918_203116.png

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

    赋值也需要先get

    简单总结:

    1. vue2 (浅响应式)
    1. vue3 (深度响应式) :

    两者的不同

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

    更多编程相关知识,请访问:编程入门!!

    以上就是vue2&vue3数据响应式原理分析及手动实现(实例详解)的详细内容,更多请关注php中文网其它相关文章!

    声明:本文转载于:CSDN,如有侵犯,请联系admin@php.cn删除
    专题推荐:vue3
    上一篇:vue的缓存有几种实现方式 下一篇:最系统的vue全套教程(详解及实例)
    php中文网线上培训班

    相关文章推荐

    • 浅析vscode中vue文件保存时怎么自动格式化• 聊聊Vue3中路由,浅析路由配置方式• vue中get和post请求的区别是什么• vue-router有几个钩子• axios是vue里面的吗

    全部评论我要评论

  • 取消发布评论发送
  • 1/1

    PHP中文网