提交小程序審覈時,有一個體驗測評,產品讓咱們根據小程序的體驗測評報告去優化小程序。前端
其中有一項是網絡請求的優化,給咱們出了很大的難題。小程序
文檔中是這樣解釋的:3分鐘之內同一個url
請求不出現兩次回包大於128KB
且如出一轍的內容api
看到這個問題的時候,首先想到的是在響應頭上加上cache-control
,通過測試發現小程序並不支持網路請求緩存。搜索發現官方明確答覆,小程序不支持網絡請求緩存:wx.request不支持http緩存緩存
既然官方不支持網絡請求緩存,那隻能本身想辦法解決這個問題了。微信
先來看一下需求:3
分鐘內,同一請求只能請求一次。網絡
分析:app
GET
請求的網絡緩存。3
分鐘,這個請求在服務端數據有沒更新。GET
請求前,先檢查本地有沒有緩存前兩點比較好實現,雖然小程序不支持網絡請求緩存,但咱們仍是能夠利用cache-control
來實現這個功能。函數
首先網絡請求需不須要情緩存統一交給服務端去作,服務端在處理GET
請求時,統一加上響應頭cache-control
,若是須要緩存就用max-age=180
,若是不須要作網絡請求就用no-cache
。前端根據響應頭信息本身作前端緩存。工具
其中的難點是前端如何知道服務端數據有沒更新,若是服務端數據更新了,前端仍是使用緩存這是有問題的。測試
通過一番思考後發現,前端提交數據後,相應的GET
請求數據會更新,也就是說前端只要有數據提交,就應該把緩存清空。
這有一個難點,當前端提交數據時,前端是不知道哪些GET
請求會所以更新數據,因此這個問題咱們沒有解決,個人方法比較粗暴:只要前端提交了數據,就將全部緩存清空。這是一個治標不治本的問題。
公司項目封裝了HTTP
請求
攔截請求,若是是GET
請求,檢查緩存,
if (request.method.toLowerCase() === "get"){ // param 請求信息 const cache = this.handleCatchControl(request) if (!cache.isRequest) return this.listener.onApiResponse(request, 200, cache.data), sequence; //將緩存返回給對應的請求 }
// param 響應頭,上下文,響應數據 this.setCatchControl(headers, context, response.data)
兩個工具函數
設置網絡請求
GET
請求緩存數據,其餘請求清空數據//若是同時發起多個`GET`請求,須要拼接以前緩存數據 ApiAgent.cacheData = Object.assign(ApiAgent.cacheData,{ [context.request.url]: { //api data, //響應數據 expireTime: Number(cacheControl.split("=")[1] + '000'), //過時時間 cacheTime: new Date().getTime(), //緩存時間 } })
// param 響應頭,上下文,響應數據 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 = {} } }
處理網絡緩存
// param 請求信息 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} }
class HandleHeaders { static get(headers: { [key: string]: string }) { const headersData: any = {} Object.keys(headers).forEach(key => { headersData[key.toLowerCase()] = headers[key] }) return headersData } }
有一點沒有說,就是這個緩存是保存在哪裏的?
既沒有用localStorage
,也沒有用globalapp
,用的是類的靜態屬性。
這樣作有3
個好處:
localStorage
數據很差清除,後期可維護性也較差globalapp
和請求無直接聯繫我在使用時遇到一個坑,是由於本身沒有理解:類能保存數據的,不能保存狀態,但類的對象是既能夠保存數據,也能夠保存狀態的。
最後,此方法仍是有很大的優化空間。
另外可添加微信ttxbg180218
交流