zoukankan      html  css  js  c++  java
  • Vue.set()和this.$set()源码解析

    前言

    我们在日常项目开发过程中,有时候我们对数组或者对象进行了一些操作后,发现页面数据没有更新到。这个时候就会有疑问,why?

    如果我们在看文档有这样一个api,以下内容:

    Vue.set()和this.$set()实现原理

    Vue.set()的源码:  ... 这里是省略的代码

    import { set } from '../observer/index'
    
    ...
    Vue.set = set
    ...

    this.$set()的源码:

    import { set } from '../observer/index'
    
    ...
    Vue.prototype.$set = set
    ...

    从上面两个源码中,我们发现Vue.set()和this.$set()这两个api的实现原理基本一模一样,都是使用了set函数。set函数是从 ../observer/index 文件中导出的,区别在于Vue.set()是将set函数绑定在Vue构造函数上,this.$set()是将set函数绑定在Vue原型上。

    接下来看下 ../observer/index 的set函数:

    function set (target: Array<any> | Object, key: any, val: any): any {
      if (process.env.NODE_ENV !== 'production' &&
        (isUndef(target) || isPrimitive(target))
      ) {
        warn(`Cannot set reactive property on undefined, null, or primitive value: ${(target: any)}`)
      }
    // 数组
    if (Array.isArray(target) && isValidArrayIndex(key)) { target.length = Math.max(target.length, key) target.splice(key, 1, val) return val }
    // 对象
    if (key in target && !(key in Object.prototype)) { target[key] = val return val } const ob = (target: any).__ob__ if (target._isVue || (ob && ob.vmCount)) { process.env.NODE_ENV !== 'production' && warn( 'Avoid adding reactive properties to a Vue instance or its root $data ' + 'at runtime - declare it upfront in the data option.' ) return val } if (!ob) { target[key] = val return val } defineReactive(ob.value, key, val) ob.dep.notify() return val }

    set函数接收三个参数分别为 target、key、val,其中target的值为数组或者对象。

    具体代码分析:

    Step1

    if (process.env.NODE_ENV !== 'production' &&
        (isUndef(target) || isPrimitive(target))
      ) {
        warn(`Cannot set reactive property on undefined, null, or primitive value: ${(target: any)}`)
      }

    isUndef和isPrimitive方法,从名字就可以看出,isUndef是判断target是不是等于undefined或者null。isPrimitive是判断target的数据类型是不是string、number、symbol、boolean中的一种。所以这里的意思是如果当前环境不是生产环境并且 isUndef(target) || isPrimitive(target) 为真的时候,那么就抛出错误警告。

    分析数组这块代码前,我们先来看下vue数组与普通js数组的区别:

     

     

    vue中的数组我们命名为arrVue,js中的普通数组命名为arrJs。我们平时调用普通数组的push、pop等方法是调用的Array原型上面定义的方法, 从图中可以看出arrJs的原型是指向Array.prototype,也就是说 arrJs.__proto__ == Array.prototype

    但是在vue的数组中,arrVue的原型其实不是指向的Array.prototype,而是指向的一个对象(我们这里给这个对象命名为arrayMethods)。arrayMethods上面只有7个push、pop等方法,并且arrayMethods的原型才是指向的Array.prototype。所以在vue中调用数组的push、pop等方法时其实不是直接调用的数组原型给我们提供的push、pop等方法,而是调用的arrayMethods给提供的push、pop等方法。vue为什么要给数组的原型链上面加上这个arrayMethods呢?这里涉及到了vue的数据响应的原理。暂时理解成vue在arrayMethods对象中做过了特殊处理,如果调用了arrayMethods提供的push、pop等7个方法,那么它会触发当前收集的依赖(这里收集的依赖可以暂时理解成渲染函数),导致页面重新渲染。换句话说,对于数组的操作,只有使用arrayMethods提供的那7个方法才会导致页面渲染,这也就解释了为什么我们使用 vueInstance.$data.arr[0] = 3;时不会导致页面出现渲染。

    Step2


    if (Array.isArray(target) && isValidArrayIndex(key)) { target.length = Math.max(target.length, key) target.splice(key, 1, val) return val }

    if判断当前target是不是数组,并且key的值是有效的数组索引。

    然后将target数组的长度设置为target.length和key中的最大值,为了防止我们传入key下标超过数组长度导致报错。

    调用arrayMethods提供的push、pop等7个方法可以导致页面重新渲染,这里使用splice是arrayMethods提供的7个方法中的一种。 

    这块代码意思是在修改数组时调用set方法时让我们能够触发响应的代码

    Step3

    if (key in target && !(key in Object.prototype)) {
        target[key] = val
        return val
      }

    如果key本来就是对象中的一个属性,并且key不是Object原型上的属性。说明这个key本来就在对象上面已经定义过了的,直接修改值就可以了,可以自动触发响应。

    const ob = (target: any).__ob__
      if (target._isVue || (ob && ob.vmCount)) {
        process.env.NODE_ENV !== 'production' && warn(
          'Avoid adding reactive properties to a Vue instance or its root $data ' +
          'at runtime - declare it upfront in the data option.'
        )
        return val
      }
      if (!ob) {
        target[key] = val
        return val
      }

    定义变量ob的值为 ```target.__ob__```,vue给响应式对象都加了一个__ob__属性,如果一个对象有这个__ob__属性,那么就说明这个对象是响应式对象,我们修改对象已有属性的时候就会触发页面渲染。 

    ```target._isVue || (ob && ob.vmCount) ```的意思是:当前的target对象是vue实例对象或者是根数据对象,那么就会抛出错误警告。

    ```if (!ob)```为真说明当前的target对象不是响应式对象,不需要响应,那么直接赋值返回即可。比如:

    ```let obj = {o: 3}; this.$set(obj1, 'o', 2);```

      defineReactive(ob.value, key, val)
      ob.dep.notify()
      return val

    这里才是vue.set()真正处理对象的地方。```defineReactive(ob.value, key, val)```的意思是给新加的属性添加依赖,以后再直接修改这个新的属性的时候就会触发页面渲染。 ```ob.dep.notify()```这句代码的意思是触发当前的依赖(这里的依赖依然可以理解成渲染函数),所以页面就会进行重新渲染。


      本文分享到这里,给朋友们推荐一个前端公众号 

  • 相关阅读:
    606. Construct String from Binary Tree 【easy】
    520. Detect Capital【easy】
    28. Implement strStr()【easy】
    521. Longest Uncommon Subsequence I【easy】
    线程,进程,任务
    nginx for windows中的一项缺陷
    nginx在windwos中的使用
    关于wxwidgets图形界面的关闭窗口的按钮无效的解决办法
    进程与线程之间的资源的关系
    关于函数可重入需要满足的条件
  • 原文地址:https://www.cnblogs.com/cczlovexw/p/13219434.html
Copyright © 2011-2022 走看看