微信小程序中網(wǎng)絡(luò)請(qǐng)求緩存的解決方法
需求
提交小程序?qū)徍藭r(shí),有一個(gè)體驗(yàn)測(cè)評(píng),產(chǎn)品讓我們根據(jù)小程序的體驗(yàn)測(cè)評(píng)報(bào)告去優(yōu)化小程序。
其中有一項(xiàng)是網(wǎng)絡(luò)請(qǐng)求的優(yōu)化,給我們出了很大的難題。
文檔中是這樣解釋的:3分鐘以內(nèi)同一個(gè)url請(qǐng)求不出現(xiàn)兩次回包大于128KB且一模一樣的內(nèi)容
看到這個(gè)問(wèn)題的時(shí)候,首先想到的是在響應(yīng)頭上加上cache-control,經(jīng)過(guò)測(cè)試發(fā)現(xiàn)小程序并不支持網(wǎng)路請(qǐng)求緩存。搜索發(fā)現(xiàn)官方明確答復(fù),小程序不支持網(wǎng)絡(luò)請(qǐng)求緩存:wx.request不支持http緩存
既然官方不支持網(wǎng)絡(luò)請(qǐng)求緩存,那只能自己想辦法解決這個(gè)問(wèn)題了。
先來(lái)看一下需求:3分鐘內(nèi),同一請(qǐng)求只能請(qǐng)求一次。
分析
分析:
- 只需做GET請(qǐng)求的網(wǎng)絡(luò)緩存。
- 緩存時(shí)間如何控制。
- 做了緩存之后,如何知道3分鐘,這個(gè)請(qǐng)求在服務(wù)端數(shù)據(jù)有沒(méi)更新。
- 提交GET請(qǐng)求前,先檢查本地有沒(méi)有緩存
前兩點(diǎn)比較好實(shí)現(xiàn),雖然小程序不支持網(wǎng)絡(luò)請(qǐng)求緩存,但我們還是可以利用cache-control來(lái)實(shí)現(xiàn)這個(gè)功能。
首先網(wǎng)絡(luò)請(qǐng)求需不需要情緩存統(tǒng)一交給服務(wù)端去做,服務(wù)端在處理GET請(qǐng)求時(shí),統(tǒng)一加上響應(yīng)頭cache-control,如果需要緩存就用max-age=180,如果不需要做網(wǎng)絡(luò)請(qǐng)求就用no-cache。前端根據(jù)響應(yīng)頭信息自己做前端緩存。
其中的難點(diǎn)是前端如何知道服務(wù)端數(shù)據(jù)有沒(méi)更新,如果服務(wù)端數(shù)據(jù)更新了,前端還是使用緩存這是有問(wèn)題的。
經(jīng)過(guò)一番思考后發(fā)現(xiàn),前端提交數(shù)據(jù)后,相應(yīng)的GET請(qǐng)求數(shù)據(jù)會(huì)更新,也就是說(shuō)前端只要有數(shù)據(jù)提交,就應(yīng)該把緩存清空。
這有一個(gè)難點(diǎn),當(dāng)前端提交數(shù)據(jù)時(shí),前端是不知道哪些GET請(qǐng)求會(huì)因此更新數(shù)據(jù),所以這個(gè)問(wèn)題我們沒(méi)有解決,我的方法比較粗暴:只要前端提交了數(shù)據(jù),就將所有緩存清空。這是一個(gè)治標(biāo)不治本的問(wèn)題。
實(shí)現(xiàn)
公司項(xiàng)目封裝了HTTP請(qǐng)求
攔截請(qǐng)求,如果是GET請(qǐng)求,檢查緩存,
- 如果緩存沒(méi)過(guò)期,將緩存返回出去,不再發(fā)請(qǐng)求
- 如果緩存過(guò)期,發(fā)請(qǐng)求
if (request.method.toLowerCase() === "get"){ // param 請(qǐng)求信息 const cache = this.handleCatchControl(request) if (!cache.isRequest) return this.listener.onApiResponse(request, 200, cache.data), sequence; //將緩存返回給對(duì)應(yīng)的請(qǐng)求 }
緩存網(wǎng)絡(luò)請(qǐng)求
// param 響應(yīng)頭,上下文,響應(yīng)數(shù)據(jù) this.setCatchControl(headers, context, response.data)
兩個(gè)工具函數(shù)
- 處理網(wǎng)絡(luò)緩存
- 設(shè)置網(wǎng)絡(luò)緩存
設(shè)置網(wǎng)絡(luò)請(qǐng)求
- GET請(qǐng)求緩存數(shù)據(jù),其他請(qǐng)求清空數(shù)據(jù)
- 數(shù)據(jù)格式:
//如果同時(shí)發(fā)起多個(gè)`GET`請(qǐng)求,需要拼接之前緩存數(shù)據(jù) ApiAgent.cacheData = Object.assign(ApiAgent.cacheData,{ [context.request.url]: { //api data, //響應(yīng)數(shù)據(jù) expireTime: Number(cacheControl.split("=")[1] + '000'), //過(guò)期時(shí)間 cacheTime: new Date().getTime(), //緩存時(shí)間 } })
// param 響應(yīng)頭,上下文,響應(yīng)數(shù)據(jù) setCatchControl(responseHeader: any, context: any, data: any) { if (context.request.method.toLowerCase() === "get") { const headers = HandleHeaders.get(responseHeader) const cacheControl = headers["cache-control"] if (cacheControl && cacheControl !== "no-cache") { ApiAgent.cacheData = Object.assign(ApiAgent.cacheData,{ [context.request.url]: { data, expireTime: Number(cacheControl.split("=")[1] + '000'), cacheTime: new Date().getTime(), } }) } } else { ApiAgent.cacheData = {} } }
處理網(wǎng)絡(luò)緩存
- 判斷緩存是否存在
- 判斷緩存有沒(méi)過(guò)期,在設(shè)置緩存時(shí),比對(duì)當(dāng)前時(shí)間和緩存時(shí)間,是否小于失效時(shí)間
// param 請(qǐng)求信息 handleCatchControl(request): any { const cacheArr = ApiAgent.cacheData if (Object.keys(cacheArr).length === 0) return { isRequest: true } let cache = {} Object.keys(cacheArr).forEach(cacheArrKey => { if (cacheArrKey === request.url) { cache = cacheArr[cacheArrKey] } }) const newDate = new Date().getTime() if (newDate - cache.cacheTime < expireTime){ return { isRequest: false, data: cache.data } } return { isRequest: true} }
響應(yīng)頭全部變成小寫(xiě),在小程序中,無(wú)法確定響應(yīng)頭的大小寫(xiě)會(huì)導(dǎo)致報(bào)錯(cuò),所以統(tǒng)一處理響應(yīng)頭
class HandleHeaders { static get(headers: { [key: string]: string }) { const headersData: any = {} Object.keys(headers).forEach(key => { headersData[key.toLowerCase()] = headers[key] }) return headersData } }
總結(jié)
有一點(diǎn)沒(méi)有說(shuō),就是這個(gè)緩存是保存在哪里的?
既沒(méi)有用localStorage,也沒(méi)有用globalapp,用的是類的靜態(tài)屬性。
這樣做有3個(gè)好處:
- 使用localStorage數(shù)據(jù)不好清除,后期可維護(hù)性也較差
- 緩存掛在globalapp和請(qǐng)求無(wú)直接聯(lián)系
- 無(wú)需在退出小程序時(shí)手動(dòng)清理緩存
我在使用時(shí)遇到一個(gè)坑,是因?yàn)樽约簺](méi)有理解:類能保存數(shù)據(jù)的,不能保存狀態(tài),但類的對(duì)象是既可以保存數(shù)據(jù),也可以保存狀態(tài)的。
最后,此方法還是有很大的優(yōu)化空間。
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)我們的支持。
上一篇:vue 中 elment-ui table合并上下兩行相同數(shù)據(jù)單元格
欄 目:JavaScript
下一篇:uni-app 組件里面獲取元素寬高的實(shí)現(xiàn)
本文標(biāo)題:微信小程序中網(wǎng)絡(luò)請(qǐng)求緩存的解決方法
本文地址:http://mengdiqiu.com.cn/a1/JavaScript/9366.html
您可能感興趣的文章
- 04-02java吃豆人代碼 js吃豆人
- 04-02微信挑一挑java代碼 微信挑一挑java代碼怎么做
- 01-10小程序簡(jiǎn)單兩欄瀑布流效果的實(shí)現(xiàn)
- 01-10微信小程序批量上傳圖片到七牛(推薦)
- 01-10微信小程序跨頁(yè)面數(shù)據(jù)傳遞事件響應(yīng)實(shí)現(xiàn)過(guò)程解析
- 01-10微信小程序按順序同步執(zhí)行的兩種方式
- 01-10ES6常用小技巧總結(jié)【去重、交換、合并、反轉(zhuǎn)、迭代、計(jì)算等】
- 01-10微信小程序?qū)崿F(xiàn)簽字功能
- 01-10JS實(shí)現(xiàn)關(guān)閉小廣告特效
- 01-10javascript實(shí)現(xiàn)點(diǎn)擊星星小游戲


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02javascript點(diǎn)線,點(diǎn)線的代碼
- 04-02javascript潛力,javascript強(qiáng)大嗎
- 04-02javascript替換字符串,js字符串的替換
- 04-02javascript移出,js 移入移出
- 04-02包含javascript舍的詞條
- 04-02javascript并行,深入理解并行編程 豆瓣
- 04-02javascript匿名,js匿名方法
- 04-02javascript警報(bào),JavaScript警告
- 04-02javascript遮蓋,JavaScript遮蓋PC端頁(yè)面
- 04-02javascript前身,javascript的前身
隨機(jī)閱讀
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 04-02jquery與jsp,用jquery
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-11Mac OSX 打開(kāi)原生自帶讀寫(xiě)NTFS功能(圖文
- 01-10C#中split用法實(shí)例總結(jié)