推广

Vue 核心之数据劫持

iseeyu2年前 (2024-02-21)推广122

在这里插入图片描述

上面的这个例子可以看出,我们完全可以控制对象属性的设置和读取。
在Vue中,作者在很多地方都非常巧妙的运用了Object.defineProperty这个方法。

vue原理:

1.监听对象属性的变化

这个应该是Vue敲开数据绑定的前大门,它通过observe(观察)每个对象的属性,添加到订阅器dep中,当数据发生变化的时候发出一个notice(预告)。 相关源代码如下:(作者采用的是ES6+flow写的,代码在src/core/observer/index.js模块里面)。

export function defineReactive (
  obj: Object,
  key: string,
  val: any,
  customSetter?: Function
) {
  const dep = new Dep()//创建订阅对象

  const property = Object.getOwnPropertyDescriptor(obj, key)//获取obj对象的key属性的描述
  //属性的描述特性里面如果configurable为false则属性的任何修改将无效
  if (property && property.configurable === false) {
    return
  }

  // cater for pre-defined getter/setters
  const getter = property && property.get
  const setter = property && property.set

  let childOb = observe(val)//创建一个观察者对象
  Object.defineProperty(obj, key, {
    enumerable: true,//可枚举
    configurable: true,//可修改
    get: function reactiveGetter () {
      const value = getter ? getter.call(obj) : val//先调用默认的get方法取值
      //这里就劫持了get方法,也是作者一个巧妙设计,在创建watcher实例的时候,通过调用对象的get方法往订阅器dep上添加这个创建的watcher实例
      if (Dep.target) {
        dep.depend()
        if (childOb) {
          childOb.dep.depend()
        }
        if (Array.isArray(value)) {
          dependArray(value)
        }
      }
      return value//返回属性值
    },
    set: function reactiveSetter (newVal) {
      const value = getter ? getter.call(obj) : val//先取旧值
      if (newVal === value) {
        return
      }
      //这个是用来判断生产环境的,可以无视
      if (process.env.NODE_ENV !== 'production' && customSetter) {
        customSetter()
      }
      if (setter) {
        setter.call(obj, newVal)
      } else {
        val = newVal
      }
      childOb = observe(newVal)//继续监听新的属性值
      dep.notify()//这个是真正劫持的目的,要对订阅者发通知了
    }
  })
}

以上是Vue监听对象属性的变化,那么题来了,我们经常在传递数据的时候往往不是一个对象,很有可能是一个数组,那是不是就没有办法了呢,答案显然是否则的。那么下面就看看作者是如何监听数组的变化:

监听数组的变化

我们还看先看这段源码:

const arrayProto = Array.prototype//原生Array的原型
export const arrayMethods = Object.create(arrayProto)

;[
  'push',
  'pop',
  'shift',
  'unshift',
  'splice',
  'sort',
  'reverse'
]
.forEach(function (method) {
  const original = arrayProto[method]//缓存元素数组原型
  //这里重写了数组的几个原型方法
  def(arrayMethods, method, function mutator () {
    //这里备份一份参数应该是从性能方面的考虑
    let i = arguments.length
    const args = new Array(i)
    while (i--) {
      args[i] = arguments[i]
    }
    const result = original.apply(this, args)//原始方法求值
    const ob = this.__ob__//这里this.__ob__指向的是数据的Observer
    let inserted
    switch (method) {
      case 'push':
        inserted = args
        break
      case 'unshift':
        inserted = args
        break
      case 'splice':
        inserted = args.slice(2)
        break
    }
    if (inserted) ob.observeArray(inserted)
    // notify change
    ob.dep.notify()
    return result
  })
})

...
//定义属性
function def (obj, key, val, enumerable) {
  Object.defineProperty(obj, key, {
    value: val,
    enumerable: !!enumerable,
    writable: true,
    configurable: true
  });
}

上面的代码主要是继承了Array本身的原型方法,然后又做了劫持修改,可以发出通知。Vue在observer数据阶段会判断如果是数组的话,则修改数组的原型,这样的话,后面对数组的任何操作都可以在劫持的过程中控制。结合Vue的思想,我们简单的写个小demo方便更好的理解:

var arrayMethod = Object.create(Array.prototype);
['push','shift'].forEach(function(method){
    Object.defineProperty(arrayMethod,method,{
        value:function(){
            var i = arguments.length
            var args = new Array(i)
            while (i--) {
              args[i] = arguments[i]
            }
            var original = Array.prototype[method];
            var result = original.apply(this,args);
            console.log("已经控制了,哈哈");
            return result;
        },
        enumerable: true,
        writable: true,
        configurable: true
    })
})
var bar = [1,2];
bar.__proto__ = arrayMethod;
bar.push(3);//控制台会打印出 “已经控制了,哈哈”;并且bar里面已经成功的添加了成员 ‘3’ 

整个过程看起来好像没有什么问题,似乎Vue已经做到了完美,其实不然,Vue还是不能检测到数据项和数组长度改变的变化,例如下面的调用:

vm.items[index] = "xxx";
vm.items.length = 100;

我们尽量避免这样的调用方式,如果确实需要,作者也帮我们实现了一个$set操作,这里就不做介绍了。

实现对象属性代理

正常情况下我们是这样实例化一个Vue对象:

var VM = new Vue({
    data:{
        name:'lhl'
    },
    el:'#id'
})

按理说我们操作数据的时候应该是VM.data.name = ‘hxx’才对,但是作者觉得这样不够简洁,所以又通过代理的方式实现了VM.name = ‘hxx’的可能。 相关代码如下:

function proxy (vm, key) {
  if (!isReserved(key)) {
    Object.defineProperty(vm, key, {
      configurable: true,
      enumerable: true,
      get: function proxyGetter () {
        return vm._data[key]
      },
      set: function proxySetter (val) {
        vm._data[key] = val;
      }
    });
  }
}

表面上看起来我们是在操作VM.name,实际上还是通过Object.defineProperty()中的get和set方法劫持实现的。

总结
Vue框架很好的利用了Object.defineProperty()这个方法来实现了数据的双向绑定,同时也达到了很好的模块间解耦

扫描二维码推送至手机访问。

版权声明:本文由西安泽虎代运营发布,如需转载请注明出处。

转载请注明出处https://www.0291.com.cn/post/57484.html

相关文章

如何在网站中给用户塑造舒适的浏览环境。

如何在网站中给用户塑造舒适的浏览环境。

相信绝大多数的企业进行营销型网站建设,其目的都是为了能够更好的营销企业产品,扩大企业的产品销量,提高企业的盈利能力。当然要实现这样的目的,就必须要确保网站能够让用户喜欢和认可企业的网站,这样才能形成强有力的转化,而要让用户喜欢和认可网站,那么网站就必须要为用户提供舒适的浏览环境,让用户能够保持愉悦的...

拼多多如何一起购买多种商品?揭秘省钱又省力的购物秘诀!

拼多多如何一起购买多种商品?揭秘省钱又省力的购物秘诀!

随着电商平台的发展,越来越多的人选择在网上购物。而拼多多作为其中的佼佼者,更是凭借其独特的拼团模式吸引了大量消费者。那么,拼多多如何一起购买多种商品呢?本文将为你揭秘省钱又省力的购物秘诀! 一、拼多多购物的优势 拼多多的购物模式主要有两大优势,一是价格优势,二是商品种类优势。拼多多...

在百度做SEO推广效果怎么样。

在百度做SEO推广效果怎么样。

1、首先分析网民获取信息的途径似乎对传统媒体来说是更坏的消息:2019年11月Edelman的调查发现在线搜索引擎超过传统媒体,成为全球最受信任的媒体资源。Edelman 2019年报告显示,全球64%的网民认为在线搜索引擎提供的是最值得信任的信息,去年这一比例是63%。同时,传统媒体信任度从去年的...

怎样优化淘宝关键词排名(淘宝怎么提高关键词搜索排名)

怎样优化淘宝关键词排名(淘宝怎么提高关键词搜索排名)

首先,我们在选词的时候,要注意产品是不是品牌产品,如果不是,就可以将品牌词去掉,其次,要看产品和词的相关性,将不相关的词去掉,再次,要看词的人气,人气太低的词不能要。...

搜索引擎SEO优化的一些经验。

搜索引擎SEO优化的一些经验。

我们会发现一个现象,很多网站无论是收录还是流量基本都来自。但我们都知道,百度的市场份额之前我看到的数据是77%-80%左右,按理来说,搜索流量里面百度占了80%-85%是正常的,可是有些网站,和搜狗收录很差,甚至为0,流量也几乎没有。 我之前看到一个网站,很老的站,百度收录1800,搜狗30,36...

用户运营,你不知道的4大运营干货

本篇文章主要是针对用户运营拉新方面的使用的运营行为(运营手段或运营),针对这一方面有许多人不知道的技巧,运营无论是从没有用户到有用户还是用户都是死亡状态都会有对应技巧应用方式。 目前针对运营拉新方面我个人总结了3大方面: 1.产品内部技巧 2.产品外部技巧 3.人员内部技...

现在,非常期待与您的又一次邂逅

我们努力让每一部企业宣传片和抖音短视频成为商业大片