Vue3源码解析(computed-计算属性)

exportfunctioncomputed(getter: ComputedGetter):ComputedRefexportfunctioncomputed( options: WritableComputedOptions):WritableComputedRefexportfunctioncomputed( getterOrOptions: ComputedGetter | WritableComputedOptions){letgetter:ComputedGetterletsetter:ComputedSetterif(isFunction(getterOrOptions)){getter=getterOrOptionssetter=NOOP}else{ getter = getterOrOptions.get setter = getterOrOptions.set }returnnewComputedRefImpl( getter, setter, isFunction(getterOrOptions) || !getterOrOptions.set )asany}

在最开始使用函数重载的方式允许computed函数接受两种类型的参数:第一种是一个getter函数, 第二种是一个带getset的对象。

接下就是在函数内部根据传入的不同类型的参数初始化函数内部的getter和setter函数,如果传入的是一个函数类型的参数,那么getter就是这个函数,setter就是一个空的操作,如果传入的参数是一个对象,则getter就等于这个对象的get函数,setter就等于这个对象的set函数。

在函数的结尾返回了一个new ComputedRefImpl,并将前面我们标准化后的参数传递给了这个构造函数。

下面我们就来分析一下ComputedRefImpl这个构造函数。

ComputedRefImpl

classComputedRefImpl{// 缓存结果private _value!: T// 重新计算开关private _dirty =truepublic readonly effect: ReactiveEffect  public readonly __v_isRef =true;  public readonly [ReactiveFlags.IS_READONLY]: booleanconstructor(    getter: ComputedGetter,

    private readonly _setter: ComputedSetter,

    isReadonly: boolean

  ){// 对传入的getter函数进行包装this.effect = effect(getter, {lazy:true,// 调度执行scheduler:() =>{if(!this._dirty) {this._dirty =true// 派发通知trigger(toRaw(this), TriggerOpTypes.SET,'value')        }      }    })  }// 访问计算属性的时候 默认调用此时的get函数getvalue() {// 是否需要重新计算if(this._dirty) {this._value =this.effect()this._dirty =false}// 访问的时候进行依赖收集 此时收集的是访问这个计算属性的副作用函数track(toRaw(this), TrackOpTypes.GET,'value')returnthis._value  }setvalue(newValue: T) {this._setter(newValue)  }}

ComputedRefImpl类在内部维护了_value和_dirty这两个非常重要的私有属性,其中_value使用用来缓存我们计算的结果,_dirty是用来控制是否需要重现计算。接下来我们来看一下这个函数的内部运行机制。

首先构造函数在初始化的时候使用了effect函数对传入getter进行了一层包装(上一篇文章中我们分析过effect函数的作用就是将传入的函数变成可响应式的副作用函数),但是这里我们在effect中传入了一些配置参数,还记得前面我们分析trigger函数的时候有这一段代码:

construn =(effect: ReactiveEffect) =>{if(effect.options.scheduler) {      effect.options.scheduler(effect)    }else{      effect()    }  }effects.forEach(run)

当属性值发生改变之后,会触发trigger函数进行派发更新,将所有依赖这个属性的effect函数循环遍历,使用run函数执行effect,如果effect的参数中配置了scheduler,则就执行scheduler函数,而不是执行依赖的副作用函数。当计算属性依赖的属性发生变化的时候,回执行包装getter函数的effect, 但是因为配置了scheduler函数,所以真正执行的是scheduler函数,在scheduler函数中并没有执行计算属性的getter函数求取新值,而是将_dirty设置为false,然后通知依赖计算属性的副作用函数进行更新, 当依赖计算属性的副作用函数收到通知的时候就会访问计算属性的get函数,此时会根据_dirty值来确定是否需要重新计算。

回到我们的这个构造函数中,只需要记得我们在构造函数初始化三个重要的点:第一:对传入的getter函数使用effect函数进行包装。第二:在使用effect包装的过程中,我们会执行getter函数,此时getter函数执行过程中对于访问到的属性会将当前的这个计算属性收集到对应的依赖集合中, 第三:传入了配置参数lazy和scheduler,这些配置参数在当前的这个计算属性所订阅的属性发生改变的时候,用来控制计算属性的调度时机。

接着我们继续分析get value,当我们访问计算属性的值时候实际上访问的就是这个函数的返回值, 它会根据_dirty的值来判断是否需要重新计算getter函数,_dirty为true需要重新执行effect函数,并将effect的值置为false,否则就返回之前缓存的_value值。在访问计算属性值的阶段会调用track函数进行依赖收集,此时收集的是访问计算属性值的副作用函数, key始终是vlaue。

最后就是当设置计算属性的值的时候会执行set函数,然后调用我们传入的_setter函数。

示例流程

至此计算属性的执行流程就分析完毕了,我们来结合一个示例来完整的过一遍整个流程:

龙华大道1号 http://www.kinghill.cn/Dynamics/2106.html

add

计算属性:{{computedData}}

import{ ref, watch,reactive, computed }from'vue'import{ effect }from'@vue/reactivity'exportdefault{name:'App',setup(){consttestData = ref(1)constcomputedData = computed(() =>{returntestData.value++    })functionaddNum(){      testData.value +=10}return{      addNum,      computedData    }  },}

你可能感兴趣的:(Vue3源码解析(computed-计算属性))