做者:秦志英javascript
上一篇文章中咱们分析了Vue3响应式的整个流程,本篇文章咱们将分析Vue3中的computed计算属性是如何实现的。html
在Vue2中咱们已经对计算属性了解的很清楚了,在Vue3中提供了一个computed
的函数做为计算属性的API,下面咱们来经过源码的角度去分析计算属性的运行流程。前端
export function computed<T>(getter: ComputedGetter<T>): ComputedRef<T> export function computed<T>( options: WritableComputedOptions<T> ): WritableComputedRef<T> export function computed<T>( getterOrOptions: ComputedGetter<T> | WritableComputedOptions<T> ) { let getter: ComputedGetter<T> let setter: ComputedSetter<T> if (isFunction(getterOrOptions)) { getter = getterOrOptions setter = NOOP } else { getter = getterOrOptions.get setter = getterOrOptions.set } return new ComputedRefImpl( getter, setter, isFunction(getterOrOptions) || !getterOrOptions.set ) as any }
computed
函数接受两种类型的参数:第一种是一个getter函数, 第二种是一个带get和set的对象。getter
和setter
函数,若是传入的是一个函数类型的参数,那么getter就是这个函数,setter就是一个空的操做,若是传入的参数是一个对象,则getter就等于这个对象的get函数,setter就等于这个对象的set函数。new ComputedRefImpl
,并将前面咱们标准化后的参数传递给了这个构造函数。ComputedRefImpl
这个构造函数。class ComputedRefImpl<T> { // 缓存结果 private _value!: T // 从新计算开关 private _dirty = true public readonly effect: ReactiveEffect<T> public readonly __v_isRef = true; public readonly [ReactiveFlags.IS_READONLY]: boolean constructor( getter: ComputedGetter<T>, private readonly _setter: ComputedSetter<T>, isReadonly: boolean ) { // 对传入的getter函数进行包装 this.effect = effect(getter, { lazy: true, // 调度执行 scheduler: () => { if (!this._dirty) { this._dirty = true // 派发通知 trigger(toRaw(this), TriggerOpTypes.SET, 'value') } } }) } // 访问计算属性的时候 默认调用此时的get函数 get value() { // 是否须要从新计算 if (this._dirty) { this._value = this.effect() this._dirty = false } // 访问的时候进行依赖收集 此时收集的是访问这个计算属性的反作用函数 track(toRaw(this), TrackOpTypes.GET, 'value') return this._value } set value(newValue: T) { this._setter(newValue) } }
ComputedRefImpl类在内部维护了_value
和_dirty
这两个很是重要的私有属性,其中_value
使用用来缓存咱们计算的结果,_dirty
是用来控制是否须要重现计算。接下来咱们来看一下这个函数的内部运行机制。vue
effect
函数对传入getter
进行了一层包装(上一篇文章中咱们分析过effect函数的做用就是将传入的函数变成可响应式的反作用函数)
,可是这里咱们在effect中传入了一些配置参数,还记得前面咱们分析trigger函数的时候有这一段代码:const run = (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
值来肯定是否须要从新计算。java
回到咱们的这个构造函数中,只须要记得咱们在构造函数初始化三个重要的点:第一:对传入的getter函数使用effect函数进行包装。第二:在使用effect包装的过程当中,咱们会执行getter函数,此时getter函数执行过程当中对于访问到的属性会将当前的这个计算属性收集到对应的依赖集合中, 第三:传入了配置参数lazy
和scheduler
,这些配置参数在当前的这个计算属性所订阅的属性发生改变的时候,用来控制计算属性的调度时机。react
get value
,当咱们访问计算属性的值时候实际上访问的就是这个函数的返回值, 它会根据_dirty
的值来判断是否须要从新计算getter函数,_dirty
为true须要从新执行effect函数,并将effect
的值置为false,不然就返回以前缓存的_value
值。在访问计算属性值的阶段会调用track
函数进行依赖收集,此时收集的是访问计算属性值的反作用函数, key始终是vlaue。_setter
函数。至此计算属性的执行流程就分析完毕了,咱们来结合一个示例来完整的过一遍整个流程:git
<template> <div> <button @click="addNum">add</button> <p>计算属性:{{computedData}}</p> </div> </template> <script> import { ref, watch,reactive, computed } from 'vue' import { effect } from '@vue/reactivity' export default { name: 'App', setup(){ const testData = ref(1) const computedData = computed(() => { return testData.value++ }) function addNum(){ testData.value += 10 } return { addNum, computedData } }, } </script>
下面是一张流程图,当点击页面中的按钮改变testData的value值时,发生的变化流程就是下面的红线部分。
github
scheduler
,因此执行的是scheduler
函数,scheduler
函数并无当即执行getter函数进行从新计算,而是将ComputedRefImpl
类内部的私有变量_dirty
设置为true,而后通知订阅当前计算属性的反作用函数进行更新操做。get value
函数,函数内部会根据_dirty值来判断是否须要从新计算,因为前面的scheduler
函数将_dirty设置为true因此此时会调用getter函数的反作用函数effect,这个时候才会从新计算并将结果返回,页面数据更新。计算属性两个最大的特色就是typescript
_value
中的值。对 Electron 感兴趣?请关注咱们的开源项目 Electron Playground,带你极速上手 Electron。segmentfault
咱们每周五会精选一些有意思的文章和消息和你们分享,来掘金关注咱们的 晓前端周刊。
咱们是好将来 · 晓黑板前端技术团队。
咱们会常常与你们分享最新最酷的行业技术知识。
欢迎来 知乎、掘金、Segmentfault、CSDN、简书、开源中国、博客园 关注咱们。