vue.js設(shè)計與實(shí)現(xiàn)-響應(yīng)系統(tǒng)-computed和lazy

我們給effect函數(shù)加入lazy屬性职辅,來實(shí)現(xiàn)懶執(zhí)行厌均,在有些場景下唬滑,我們并不希望它立即執(zhí)行,而是希望在需要的時候才執(zhí)行棺弊,例如計算屬性间雀,我們在options里面添加lazy屬性來達(dá)到目的,即如下:

effect(
    ()=>{console.log(obj.foo)},
    {
        lazy:true
    }
)

lazy選項和之前介紹的scheduler一樣镊屎,它通過options選項對象指定惹挟,有了它,我們就可以修改effect函數(shù)實(shí)現(xiàn)邏輯缝驳,當(dāng)options.lazy為true時连锯,則不立即執(zhí)行副作用函數(shù):

 function effect(fn,options={}){
        const effectFn=()=>{
            cleanup(effectFn)
            activeEffect = effectFn
            effectStack.push(effectFn)
            const res = fn()
            effectStack.pop()
            activeEffect = effectStack[effectStack.length-1]
            return res
        }
        effectFn.deps=[]
        effectFn.options = options
        if(!options.lazy){
            effectFn()
        }
        return effectFn
    }

我們想實(shí)現(xiàn)通過定義computed函數(shù)來實(shí)現(xiàn)計算,實(shí)現(xiàn)方法如下:

function computed(getter){
    let value
    // dirty用來標(biāo)識是否重新計算值
    let dirty = true
    const effectFn=effect(
        getter,
        {
            lazy:true,
            // 屬性值發(fā)生變化需要dirty值變?yōu)閠rue用狱,重新計算
            scheduler(){
                dirty = true
            }
        }
    )
    const obj = {
        get value(){
            if(dirty){
                value = effectFn()
                // 將dirty設(shè)置為false运怖,下一次訪問直接使用緩存到value中的值
                dirty = false
            }
            return value
        }
    }
    return obj
}

使用時:

const sumRes = computed(()=>obj.bar+obj.foo)
console.log(sumRes.value)

但假如我們在外面這樣使用:

const sumRes = computed(()=>obj.bar+obj.foo)
effect(()=>{
    console.log(sumRes.value)
})
obj.foo++
console.log(sumRes.value)

那么effect函數(shù)里還是打印2

原因:
從本質(zhì)上看這就是一個典型的 effect 嵌套。一個計算屬性內(nèi)部擁有自己的 effect夏伊,并且它是懶執(zhí)行的摇展,只有當(dāng)真正讀取計算屬性的值時才會執(zhí)行。對于計算屬性的 getter 函數(shù)來說溺忧,它里面訪問的響應(yīng)式數(shù)據(jù)只會把computed 內(nèi)部的 effect 收集為依賴咏连。而當(dāng)把計算屬性用于另外一個 effect 時盯孙,就會發(fā)生 effect 嵌套,外層的 effect 不會被內(nèi)層 effect 中的響應(yīng)式數(shù)據(jù)收集祟滴。

解決辦法:

我們可以手動調(diào)用 track 函數(shù)進(jìn)行追蹤振惰;當(dāng)計算屬性依賴的響應(yīng)式數(shù)據(jù)發(fā)生變化時,我們可以手動調(diào)用trigger 函數(shù)觸發(fā)響應(yīng):

function computed(getter){
        let value
        let dirty = true
        const effectFn=effect(
            getter,
            {
                lazy:true,
                scheduler(){
                    if(!dirty){
                        dirty = true
                        // 當(dāng)計算屬性依賴響應(yīng)式數(shù)據(jù)變化時垄懂,手動調(diào)用trigger函數(shù)
                        trigger(obj,'value')
                    }
                }
            }
        )
        const obj = {
            get value(){
                if(dirty){
                    value = effectFn()
                    dirty = false
                }
                // 當(dāng)讀取value時骑晶,手動調(diào)用track函數(shù)追蹤
                track(obj,'value')
                return value
            }
        }
        return obj
    }

最后整體代碼如下:可自行測試

const data={foo:1,bar:1}
    let activeEffect;
    const bucket = new WeakMap()
    const effectStack = []
    function effect(fn,options={}){
        const effectFn=()=>{
            cleanup(effectFn)
            activeEffect = effectFn
            effectStack.push(effectFn)
            const res = fn()
            effectStack.pop()
            activeEffect = effectStack[effectStack.length-1]
            return res
        }
        effectFn.deps=[]
        effectFn.options = options
        if(!options.lazy){
            effectFn()
        }
        return effectFn
    }
    function cleanup(effectFn){
        for(let i=0;i<effectFn.deps.length;i++){
            effectFn.deps[i].delete(effectFn)
        }
        effectFn.deps.length = 0
    }
    function track(target,key){
        if(!activeEffect)return
        let depsMap = bucket.get(target)
        if(!depsMap){bucket.set(target,depsMap=new Map())}
        let deps = depsMap.get(key)
        if(!deps){depsMap.set(key,deps=new Set())}
        deps.add(activeEffect)
        activeEffect.deps.push(deps)
    }
    function trigger(target,key){
        let depsMap = bucket.get(target)
        if(!depsMap) return
        const effects = depsMap.get(key)
        const effectsToRun = new Set()
        effects && effects.forEach(effectFn=>{
            if(effectFn!==activeEffect){
                effectsToRun.add(effectFn)
            }
        })
        // 調(diào)度執(zhí)行
        effectsToRun.forEach(effectFn=>{
            if(effectFn.options.scheduler){
                effectFn.options.scheduler(effectFn)
            }else{
                effectFn()
            }
        })
    }
    const obj = new Proxy(data,{
        get(target,key){
            track(target,key)
            return target[key]
        },
        set(target,key,newVal){
            target[key] = newVal
            trigger(target,key)
        }
    })
    
    function computed(getter){
        let value
        let dirty = true
        const effectFn=effect(
            getter,
            {
                lazy:true,
                scheduler(){
                    if(!dirty){
                        dirty = true
                        // 當(dāng)計算屬性依賴響應(yīng)式數(shù)據(jù)變化時,手動調(diào)用trigger函數(shù)
                        trigger(obj,'value')
                    }
                }
            }
        )
        const obj = {
            get value(){
                if(dirty){
                    value = effectFn()
                    dirty = false
                }
                // 當(dāng)讀取value時草慧,手動調(diào)用track函數(shù)追蹤
                track(obj,'value')
                return value
            }
        }
        return obj
    }
    const sumRes = computed(()=>obj.bar+obj.foo)
    effect(
        ()=>{
            console.log(sumRes.value)
        }
    )
    obj.foo++
    console.log(sumRes.value)
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末桶蛔,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子漫谷,更是在濱河造成了極大的恐慌仔雷,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,248評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件抖剿,死亡現(xiàn)場離奇詭異,居然都是意外死亡识窿,警方通過查閱死者的電腦和手機(jī)斩郎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來喻频,“玉大人缩宜,你說我怎么就攤上這事∩拢” “怎么了锻煌?”我有些...
    開封第一講書人閱讀 153,443評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長姻蚓。 經(jīng)常有香客問我宋梧,道長,這世上最難降的妖魔是什么狰挡? 我笑而不...
    開封第一講書人閱讀 55,475評論 1 279
  • 正文 為了忘掉前任捂龄,我火速辦了婚禮,結(jié)果婚禮上加叁,老公的妹妹穿的比我還像新娘倦沧。我一直安慰自己,他們只是感情好它匕,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,458評論 5 374
  • 文/花漫 我一把揭開白布展融。 她就那樣靜靜地躺著,像睡著了一般豫柬。 火紅的嫁衣襯著肌膚如雪告希。 梳的紋絲不亂的頭發(fā)上扑浸,一...
    開封第一講書人閱讀 49,185評論 1 284
  • 那天,我揣著相機(jī)與錄音暂雹,去河邊找鬼首装。 笑死,一個胖子當(dāng)著我的面吹牛杭跪,可吹牛的內(nèi)容都是我干的仙逻。 我是一名探鬼主播,決...
    沈念sama閱讀 38,451評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼涧尿,長吁一口氣:“原來是場噩夢啊……” “哼系奉!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起姑廉,我...
    開封第一講書人閱讀 37,112評論 0 261
  • 序言:老撾萬榮一對情侶失蹤缺亮,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后桥言,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體萌踱,經(jīng)...
    沈念sama閱讀 43,609評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,083評論 2 325
  • 正文 我和宋清朗相戀三年号阿,在試婚紗的時候發(fā)現(xiàn)自己被綠了并鸵。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,163評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡扔涧,死狀恐怖园担,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情枯夜,我是刑警寧澤弯汰,帶...
    沈念sama閱讀 33,803評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站湖雹,受9級特大地震影響咏闪,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜摔吏,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,357評論 3 307
  • 文/蒙蒙 一汤踏、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧舔腾,春花似錦溪胶、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春才避,著一層夾襖步出監(jiān)牢的瞬間橱夭,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評論 1 261
  • 我被黑心中介騙來泰國打工桑逝, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留棘劣,地道東北人。 一個月前我還...
    沈念sama閱讀 45,636評論 2 355
  • 正文 我出身青樓楞遏,卻偏偏與公主長得像茬暇,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子寡喝,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,925評論 2 344

推薦閱讀更多精彩內(nèi)容