詳解Vue中是如何實(shí)現(xiàn)cache緩存的
這篇文章分享一個(gè)比較有意思的東西,那就是Vue
中如何實(shí)現(xiàn)cache
緩存的。提前揭曉答案--閉包。下面我們就來(lái)手寫(xiě)一個(gè)吧。第一步,毫無(wú)疑問(wèn)當(dāng)然是我們的測(cè)試文件咯。
添加測(cè)試文件
繼續(xù)使用vitest
測(cè)試,想知道具體怎么操作的,可以看下專欄第一篇文章。
剛開(kāi)始學(xué)習(xí)寫(xiě)測(cè)試,大家多提意見(jiàn)哈。
test.only("cached", () => { const fnMock = vi.fn((str) => str) const getName = cached(fnMock) const name = getName("mick") expect(name).toBe("mick") expect(fnMock).toHaveBeenCalled() expect(fnMock).toHaveBeenCalledTimes(1) const name1 = getName("mick") expect(name1).toBe("mick") expect(fnMock).toHaveBeenCalledTimes(1) })
首先模擬一個(gè)函數(shù),這個(gè)函數(shù)傳入一個(gè)字符串,返回一個(gè)字符串。為什么要這么模擬呢?我們拿Vue
當(dāng)中capitalize
首字母轉(zhuǎn)大寫(xiě)舉例(當(dāng)然我們可以暫時(shí)不用太多的關(guān)心capitalize
是怎么實(shí)現(xiàn)的)
const capitalize = cached((str) => { return str.charAt(0).toUpperCase() + str.slice(1) })
可以看到cached
傳入了一個(gè)函數(shù),這個(gè)函數(shù)要實(shí)現(xiàn)的就是傳入一個(gè)字符串,返回一個(gè)字符串(首字母轉(zhuǎn)成大寫(xiě))。所以我們要模擬的就是類似這樣的一個(gè)函數(shù)。
回歸正題
模擬了一個(gè)函數(shù)fnMock
,傳給cached
并執(zhí)行,返回了getName
函數(shù)。然后調(diào)用getName
函數(shù)傳入字符串mick
,用name
變量接收,這個(gè)變量我們期望應(yīng)該是mick
。此時(shí)模擬的fnMock
應(yīng)該被調(diào)用了一次,當(dāng)再次執(zhí)行getName
函數(shù)同樣傳入mick
時(shí),仍然能拿到這個(gè)值,但是模擬的fnMock
還是被調(diào)用一次。因?yàn)?code>mick被緩存了。
沒(méi)錯(cuò),就是用文章開(kāi)頭提到的"閉包"解決。
實(shí)現(xiàn)cache
不廢話,直接上代碼
export function cached(fn) { const cache = Object.create(null) return function cachedFn(str) { var hit = cache[str] return hit || (cache[str] = fn(str)) } }
首先創(chuàng)建一個(gè)空對(duì)象方便存儲(chǔ)需要緩存的值。然后返回一個(gè)函數(shù)cachedFn
,參數(shù)fn
就是就是我們測(cè)試中的fnMock
,函數(shù)內(nèi)部首先去獲取我們想要拿到的值cache[str]
,第一次肯定是拿不到的,所以需要對(duì)其賦值cache[str] = fn(str)
并返回,此時(shí)fnMock
已經(jīng)被執(zhí)行了1次。此時(shí)根據(jù)閉包的特性,傳入的str
已經(jīng)被緩存了起來(lái)。當(dāng)?shù)诙蝹魅胪瑯拥闹档臅r(shí)候,var hit = cache[str]
就可以獲取到了,則直接返回,所以fnMock
不會(huì)被再次執(zhí)行,只有首次執(zhí)行了1次。
到此這篇關(guān)于詳解Vue中是如何實(shí)現(xiàn)cache緩存的的文章就介紹到這了,更多相關(guān)Vue cache緩存內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue+iview tabs context-menu 彈出框修改樣式的方法
今天遇到一個(gè)需求說(shuō)頁(yè)面頂部的菜單右鍵彈出框離得有點(diǎn)遠(yuǎn),需要我們做調(diào)整,下面小編給大家分享下vue+iview tabs context-menu 彈出框修改樣式的方法,感興趣的朋友跟隨小編一起看看吧2024-06-06Vue.js實(shí)現(xiàn)的表格增加刪除demo示例
這篇文章主要介紹了Vue.js實(shí)現(xiàn)的表格增加刪除demo,結(jié)合實(shí)例形式分析了vue.js數(shù)據(jù)綁定及元素增加、刪除等相關(guān)操作技巧,需要的朋友可以參考下2018-05-05使用async-validator編寫(xiě)Form組件的方法
本篇文章主要介紹了使用 async-validator 編寫(xiě) Form 組件的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-01-01Vuex如何獲取getter對(duì)象中的值(包括module中的getter)
這篇文章主要介紹了Vuex如何獲取getter對(duì)象中的值(包括module中的getter),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-08-08快速解決vue動(dòng)態(tài)綁定多個(gè)class的官方實(shí)例語(yǔ)法無(wú)效的問(wèn)題
今天小編就為大家分享一篇快速解決vue動(dòng)態(tài)綁定多個(gè)class的官方實(shí)例語(yǔ)法無(wú)效的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-09-09