手写Vue源码之计算属性

Posted lyt0207

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了手写Vue源码之计算属性相关的知识,希望对你有一定的参考价值。

计算属性 特点: 默认不执行,等用户取值的时候再去执行,会缓存取值结果,
如果依赖的值发生了变化 会更新dirty的属性,再次取值时,可以重新求值

import {pushTarget,popTarget} from ‘./dep‘
import { utils } from ‘../utils‘
let id = 0 //watcher的唯一标识
class Watcher{ //每次产生一个watcher都要有一个唯一标识
    //vm ,msg, (newValue,oldValue)=>{}, {user:true}
    //vm, ()=>this.firstName+this.lastName  ()=>{} {lazy:true}
    constructor(vm,exprOrfn,cb = {},opts = {}){ //cb = {},opts = {} 默认值
        this.vm = vm //当前组件实例
        this.exprOrfn = exprOrfn //判断用户传入的是表达式还是函数msg
        if(typeof exprOrfn === ‘function‘) {
            this.getter = exprOrfn //getter就是传入的第二个参数
        }else { //如果是表达式就将对应得表达式取出来 赋值给getter
            this.getter = function () {
                return utils.getValue(vm,exprOrfn)
            }
        }
        if(opts.user){ //标识用户自己写的Watcher
            this.user = true
        }
        this.lazy = opts.lazy //如果这个值为true说明是计算属性
        this.dirty = this.lazy //计算属性的缓存
        this.cb = cb //用户传入的回调函数 vm.$watch(‘msg‘,cb)
        this.opts = {} //其他参数
        this.id = id++
        this.deps = []
        this.depId = new Set()
        //创建watcher得时候先将表达式对应得老值取出来,更新得时候再取新值
        this.value = this.lazy ? undefined : this.get() //如果是计算属性的话不会立即调用get方法
         

        this.get() //每次创建一个watcher就执行自己的get方法
    }
    get() {
        //Dep.target也会存放用户的watcher = user watcher
        pushTarget(this) //渲染watcher保存起来 Dep.target = watcher

        let value = this.getter.call(vm) //让当前传入的函数updataComponent执行,从实例上取值渲染页面 会调用Object.defineProperty 的get方法
        popTarget() //取完值后 清空当前watcher
        return value //返回新值
    }
    addDep(dep){ //过滤,不要重复存入watcher到dep中,也不要重复存dep。
        let id = dep.id
        if(!this.depId.has(id)) {
            this.depId.add(id)
            this.deps.push(dep) //将dep存入watcher
            dep.addSub(this) //将watcher存入dep
        }
    }
    eveluate(){
        this.value = this.get() //通过这个将计算属性watcher放进栈,并且Dep.target指向了计算属性watcher,然后走getter
        this.dirty = false //值求过了,下次渲染的时候不用求了,当计算属性的值发生变化了就会调用updata方法

    }
    depend(){
        let i = this.deps.length
        while(i--){
            this.deps[i].depend()
        }
    }
/**
 * 重要部分updata()方法
 * 加入页面有四个地方需要更新属性,那我们希望不要更新四次,而是一次性更新
 * 防止不停的更新
    把需要更新的watcher先存起来 放进一个异步队列queueWatcher,然后通过nextTick异步执行
    把重复的watcher过滤掉
    等待这轮更新完就清空队列 就是说等待主执行栈执行完了就执行异步任务,也可以理解为页面所有属性都赋值完再执行这个异步方法
 */
    updata(){ //如果立即调用get 会导致页面刷新频繁 用异步更新来解决
        if(this.lazy){
            this.dirty = true //计算属性依赖的值发生了变化,需要取值重新计算
        }else {
            // this.get()
            queueWatcher(this) //queueWatcher异步队列
        }
        
    }
    run(){
        let value = this.get() //新值
        if(this.value !== value) {
            this.cb(value,this.value)
        }
    }
    
}
//渲染watcher、计算属性、vm.$watch 都属于Watcher实例
export default Watcher

function flushQueue () {
    queue.forEach(watcher => watcher.run) //执行更新
    //清空,下次使用
    has = {}
    queue = []
}
let has ={};
let queue = []
function queueWatcher(watcher){
    let id = watcher.id
    if(has[id] == null){
        has[id] =true
        queue.push(watcher) //相同的Watcher只会存一个到queue中

        //延迟清空队列
        // setTimeout(flushQueue,0) 这个方法也可以但vue内部原理是nextTick
        nextTick(flushQueue)
    }
}
let callbacks =[] //有可能用户也会写一个nextTick方法,这时候就需要把nextTick的回调函数放进一个数组里面,再依次执行,
function nextTick(cb){
    callbacks.push(cb)

    //异步刷新这个callbacks 
    //异步任务先执行微任务在执行宏任务,微任务:Promise, mutationObserver, 宏任务:setImmediate setTimeout
    let timeFunc = () => {
        flushCallbacks()
    }
    //判断当前浏览器执行的异步方法
    if(Promise) {
        return Promise.resolve().then(timeFunc)
    }
    if(MutationObserver){ //创建并返回一个新的 MutationObserver 它会在指定的DOM发生变化时被调用
        let observe = new MutationObserver(timeFunc);
        let textNode = document.createTextNode(1)
        observe.observe(textNode,{characterData:true})
        textNode.textContent(2)
        return
    }
    if(setImmediate) {
        return setImmediate(timeFunc)
    }
    //以上方法都不支持的时候就调用setTimeout setTimeout是宏任务,会在下一轮执行 事件循环机制的相关知识,但我们希望尽量再本轮执行,所以先判断支不支持微任务
    setTimeout(timeFunc,0)
}

 

//计算属性 特点: 默认不执行,等用户取值的时候再去执行,会缓存取值结果,
//如果依赖的值发生了变化 会更新dirty的属性,再次取值时,可以重新求值
function initComputed(vm, computed) {
    let wathcers = vm.watchersComputed = Object.create(null) //创建存储计算属性的watcher对象
    for (let key in computed){
        let userDef = computed[key]
        //new Watcher 此时什么都不会做 配置了lazy和dirty
        wathcers[key] = new Watcher(vm, userDef, ()=>{}, {lazy:true}) //lazy:true表示是计算属性,默认刚开始不会执行
        Object.defineProperty(vm,key,{  //将计算属性的方法定义到vm上
            get: createComputedGetter(vm, key) //用户在vm上取值的时候会触发这个方法
        }) 
    }

} 

function createComputedGetter (vm, key) {
    let watcher = vm__watchersComputed[key] //这个watcher就是我们定义的计算属性watcher的实例
    return function () { //用户取值执行此方法
        if(watcher){
            //如果dirty是false的话 不需要重新执行计算属性中的方法
            if(watcher.dirty){ //如果页面取值,dirty为true,就会调用watcher的get方法
                watcher.eveluate()
            }
            if(Dep.target){ //watcher就是计算属性watcher
                watcher.depend() //将计算属性的watcher添加到依赖列表
            }
            return watcher.value
        }
    }
}

 重要步骤:

1.计算属性主要就是创建一个watcher,并且有个标识lazy:true,表示是计算属性watcher,首次渲染并不会执行get方法,当用户需要取值的时候才会执行get方法。
this.value = this.lazy ? undefined : this.get() //如果是计算属性的话不会立即调用get方法

 

2.首次渲染的时候会创建一个渲染watcher,并将渲染watcher放在栈里,然后执行getter,编译模板,然后要取fullName的值,会调用watcher的evaluate进行求值并将dirty改为false表示,已经取完了
eveluate(){
    this.value = this.get() //通过这个将计算属性watcher放进栈,并且Dep.target指向了计算属性watcher,然后走getter
    this.dirty = false //值求过了,下次渲染的时候不用求了,当计算属性的值发生变化了就会调用updata方法

}

 

3.接下来执行get方法将计算属性wathcer放进栈里,并让Dep.target指向这个计算属性watcher,然后执行getter,将当前依赖的值的watcher存到dep中,并且让Dep.target指向渲染watcher,
get() {
    //Dep.target也会存放用户的watcher = user watcher
    pushTarget(this) //渲染watcher保存起来 Dep.target = watcher

    let value = this.getter.call(vm) //让当前传入的函数updataComponent执行,从实例上取值渲染页面 会调用Object.defineProperty 的get方法
    popTarget() //取完值后 清空当前watcher
    return value //返回新值
}

 

4.然后建立计算属性的watcher和渲染wathcer的依赖关系,当计算属性的返回值发生了变化就会通知渲染watcher重新渲染。
depend(){
    let i = this.deps.length
    while(i--){
        this.deps[i].depend()
    }
}

 

以上是关于手写Vue源码之计算属性的主要内容,如果未能解决你的问题,请参考以下文章

Vue源码之计算属性watcher

Vue源码之渲染watcher

Vue源码之渲染watcher

手写Vuex源码

vue3源码分析——手写diff算法

vue3源码分析——手写diff算法