微信小程序中網(wǎng)絡(luò)請求緩存的解決方法
需求
提交小程序?qū)徍藭r,有一個體驗測評,產(chǎn)品讓我們根據(jù)小程序的體驗測評報告去優(yōu)化小程序。
其中有一項是網(wǎng)絡(luò)請求的優(yōu)化,給我們出了很大的難題。
文檔中是這樣解釋的:3分鐘以內(nèi)同一個url請求不出現(xiàn)兩次回包大于128KB且一模一樣的內(nèi)容
看到這個問題的時候,首先想到的是在響應(yīng)頭上加上cache-control,經(jīng)過測試發(fā)現(xiàn)小程序并不支持網(wǎng)路請求緩存。搜索發(fā)現(xiàn)官方明確答復(fù),小程序不支持網(wǎng)絡(luò)請求緩存:wx.request不支持http緩存
既然官方不支持網(wǎng)絡(luò)請求緩存,那只能自己想辦法解決這個問題了。
先來看一下需求:3分鐘內(nèi),同一請求只能請求一次。
分析
分析:
- 只需做GET請求的網(wǎng)絡(luò)緩存。
- 緩存時間如何控制。
- 做了緩存之后,如何知道3分鐘,這個請求在服務(wù)端數(shù)據(jù)有沒更新。
- 提交GET請求前,先檢查本地有沒有緩存
前兩點比較好實現(xiàn),雖然小程序不支持網(wǎng)絡(luò)請求緩存,但我們還是可以利用cache-control來實現(xiàn)這個功能。
首先網(wǎng)絡(luò)請求需不需要情緩存統(tǒng)一交給服務(wù)端去做,服務(wù)端在處理GET請求時,統(tǒng)一加上響應(yīng)頭cache-control,如果需要緩存就用max-age=180,如果不需要做網(wǎng)絡(luò)請求就用no-cache。前端根據(jù)響應(yīng)頭信息自己做前端緩存。
其中的難點是前端如何知道服務(wù)端數(shù)據(jù)有沒更新,如果服務(wù)端數(shù)據(jù)更新了,前端還是使用緩存這是有問題的。
經(jīng)過一番思考后發(fā)現(xiàn),前端提交數(shù)據(jù)后,相應(yīng)的GET請求數(shù)據(jù)會更新,也就是說前端只要有數(shù)據(jù)提交,就應(yīng)該把緩存清空。
這有一個難點,當(dāng)前端提交數(shù)據(jù)時,前端是不知道哪些GET請求會因此更新數(shù)據(jù),所以這個問題我們沒有解決,我的方法比較粗暴:只要前端提交了數(shù)據(jù),就將所有緩存清空。這是一個治標(biāo)不治本的問題。
實現(xiàn)
公司項目封裝了HTTP請求
攔截請求,如果是GET請求,檢查緩存,
- 如果緩存沒過期,將緩存返回出去,不再發(fā)請求
- 如果緩存過期,發(fā)請求
if (request.method.toLowerCase() === "get"){
// param 請求信息
const cache = this.handleCatchControl(request)
if (!cache.isRequest)
return this.listener.onApiResponse(request, 200, cache.data), sequence; //將緩存返回給對應(yīng)的請求
}
緩存網(wǎng)絡(luò)請求
// param 響應(yīng)頭,上下文,響應(yīng)數(shù)據(jù) this.setCatchControl(headers, context, response.data)
兩個工具函數(shù)
- 處理網(wǎng)絡(luò)緩存
- 設(shè)置網(wǎng)絡(luò)緩存
設(shè)置網(wǎng)絡(luò)請求
- GET請求緩存數(shù)據(jù),其他請求清空數(shù)據(jù)
- 數(shù)據(jù)格式:
//如果同時發(fā)起多個`GET`請求,需要拼接之前緩存數(shù)據(jù)
ApiAgent.cacheData = Object.assign(ApiAgent.cacheData,{
[context.request.url]: { //api
data, //響應(yīng)數(shù)據(jù)
expireTime: Number(cacheControl.split("=")[1] + '000'), //過期時間
cacheTime: new Date().getTime(), //緩存時間
}
})
// 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ò)緩存
- 判斷緩存是否存在
- 判斷緩存有沒過期,在設(shè)置緩存時,比對當(dāng)前時間和緩存時間,是否小于失效時間
// 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}
}
響應(yīng)頭全部變成小寫,在小程序中,無法確定響應(yīng)頭的大小寫會導(dǎo)致報錯,所以統(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é)
有一點沒有說,就是這個緩存是保存在哪里的?
既沒有用localStorage,也沒有用globalapp,用的是類的靜態(tài)屬性。
這樣做有3個好處:
- 使用localStorage數(shù)據(jù)不好清除,后期可維護性也較差
- 緩存掛在globalapp和請求無直接聯(lián)系
- 無需在退出小程序時手動清理緩存
我在使用時遇到一個坑,是因為自己沒有理解:類能保存數(shù)據(jù)的,不能保存狀態(tài),但類的對象是既可以保存數(shù)據(jù),也可以保存狀態(tài)的。
最后,此方法還是有很大的優(yōu)化空間。
好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
相關(guān)文章
JavaScript 申明函數(shù)的三種方法 每個函數(shù)就是一個對象(一)
JavaScript 申明函數(shù)的三種方法 每個函數(shù)就是一個對象(一)2009-12-12
javascript將浮點數(shù)轉(zhuǎn)換成整數(shù)的三個方法
將浮點數(shù)轉(zhuǎn)換成整數(shù)方法有很多,本例為大家介紹常用的三個方法,如果讀者想到其他好用方法,也可以交流一下2014-06-06
javascript延時重復(fù)執(zhí)行函數(shù) lLoopRun.js
javascript延時重復(fù)執(zhí)行函數(shù) lLoopRun.js...2007-06-06
小程序安全指南之如何禁止外部直接跳轉(zhuǎn)到小程序某頁面
由于小程序跳轉(zhuǎn)的對象比較多,各自的規(guī)則又不一樣,因此小程序跳轉(zhuǎn)外部鏈接是用戶咨詢較多的問題之一,下面這篇文章主要給大家介紹了關(guān)于小程序安全指南之如何禁止外部直接跳轉(zhuǎn)到小程序某頁面的相關(guān)資料,需要的朋友可以參考下2022-09-09

