日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關咨詢
選擇下列產品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關閉右側工具欄

新聞中心

這里有您想知道的互聯(lián)網營銷解決方案
前端api請求緩存方案

 在開發(fā) web 應用程序時,性能都是必不可少的話題。對于webpack打包的單頁面應用程序而言,我們可以采用很多方式來對性能進行優(yōu)化,比方說 tree-shaking、模塊懶加載、利用 extrens 網絡cdn 加速這些常規(guī)的優(yōu)化。

創(chuàng)新互聯(lián)擁有十年成都網站建設工作經驗,為各大企業(yè)提供成都做網站、網站建設服務,對于網頁設計、PC網站建設(電腦版網站建設)、成都app軟件開發(fā)公司、wap網站建設(手機版網站建設)、程序開發(fā)、網站優(yōu)化(SEO優(yōu)化)、微網站、國際域名空間等,憑借多年來在互聯(lián)網的打拼,我們在互聯(lián)網網站建設行業(yè)積累了很多網站制作、網站設計、網絡營銷經驗,集策劃、開發(fā)、設計、營銷、管理等網站化運作于一體,具備承接各種規(guī)模類型的網站建設項目的能力。

甚至在vue-cli 項目中我們可以使用 --modern 指令生成新舊兩份瀏覽器代碼來對程序進行優(yōu)化。

而事實上,緩存一定是提升web應用程序有效方法之一,尤其是用戶受限于網速的情況下。提升系統(tǒng)的響應能力,降低網絡的消耗。

當然,內容越接近于用戶,則緩存的速度就會越快,緩存的有效性則會越高。

以客戶端而言,我們有很多緩存數(shù)據(jù)與資源的方法,例如 標準的瀏覽器緩存 以及 目前火熱的 Service worker。

但是,他們更適合靜態(tài)內容的緩存。例如 html,js,css以及圖片等文件。而緩存系統(tǒng)數(shù)據(jù),我采用另外的方案。

那我現(xiàn)在就對我應用到項目中的各種 api 請求方案,從簡單到復雜依次介紹一下。

方案一、 數(shù)據(jù)緩存

簡單的 數(shù)據(jù) 緩存,第一次請求時候獲取數(shù)據(jù),之后便使用數(shù)據(jù),不再請求后端api。

代碼如下:

 
 
 
  1. const dataCache = new Map() 
  2. async getWares() { 
  3.     let key = 'wares' 
  4.     // 從data 緩存中獲取 數(shù)據(jù) 
  5.     let data = dataCache.get(key) 
  6.     if (!data) { 
  7.         // 沒有數(shù)據(jù)請求服務器 
  8.         const res = await request.get('/getWares') 
  9.         // 其他操作 
  10.         ... 
  11.         data = ... 
  12.         // 設置數(shù)據(jù)緩存 
  13.         dataCache.set(key, data) 
  14.     } 
  15.     return data 
  16. }

第一行代碼 使用了 es6以上的 Map,如果對map不是很理解的情況下,你可以參考

ECMAScript 6 入門 Set 和 Map 或者 Exploring ES6 關于 map 和 set的介紹,此處可以理解為一個鍵值對存儲結構。

之后 代碼 使用 了 async 函數(shù),可以將異步操作變得更為方便。你可以參考ECMAScript 6 入門 async函數(shù)來進行學習或者鞏固知識。

代碼本身很容易理解,是利用 Map 對象對數(shù)據(jù)進行緩存,之后調用從 Map 對象來取數(shù)據(jù)。對于及其簡單的業(yè)務場景,直接利用此代碼即可。

調用方式:

 
 
 
  1. getWares().then( ... ) 
  2. // 第二次調用 取得先前的data 
  3. getWares().then( ... )

方案二、 promise緩存

方案一本身是不足的。因為如果考慮同時兩個以上的調用此 api,會因為請求未返回而進行第二次請求api。

當然,如果你在系統(tǒng)中添加類似于 vuex、redux這樣的單一數(shù)據(jù)源框架,這樣的問題不太會遇到,但是有時候我們想在各個復雜組件分別調用api,而不想對組件進行組件通信數(shù)據(jù)時候,便會遇到此場景。

 
 
 
  1. const promiseCache = new Map() 
  2. getWares() { 
  3.     const key = 'wares' 
  4.     let promise = promiseCache.get(key); 
  5.     // 當前promise緩存中沒有 該promise 
  6.     if (!promise) { 
  7.         promise = request.get('/getWares').then(res => { 
  8.             // 對res 進行操作 
  9.             ... 
  10.         }).catch(error => { 
  11.             // 在請求回來后,如果出現(xiàn)問題,把promise從cache中刪除 以避免第二次請求繼續(xù)出錯S 
  12.             promiseCache.delete(key) 
  13.             return Promise.reject(error) 
  14.         }) 
  15.     } 
  16.     // 返回promise 
  17.     return promise 
  18. }

該代碼避免了方案一的同一時間多次請求的問題。同時也在后端出錯的情況下對promise進行了刪除,不會出現(xiàn)緩存了錯誤的promise就一直出錯的問題。

調用方式:

 
 
 
  1. getWares().then( ... ) 
  2. // 第二次調用 取得先前的promise 
  3. getWares().then( ... )

方案三、 多promise 緩存

該方案是同時需要 一個以上 的api請求的情況下,對數(shù)據(jù)同時返回,如果某一個api發(fā)生錯誤的情況下。

均不返回正確數(shù)據(jù)。

 
 
 
  1. const querys ={ 
  2.     wares: 'getWares', 
  3.     skus: 'getSku' 
  4. const promiseCache = new Map() 
  5. async queryAll(queryApiName) { 
  6.     // 判斷傳入的數(shù)據(jù)是否是數(shù)組 
  7.     const queryIsArray = Array.isArray(queryApiName) 
  8.     // 統(tǒng)一化處理數(shù)據(jù),無論是字符串還是數(shù)組均視為數(shù)組 
  9.     const apis = queryIsArray ? queryApiName : [queryApiName] 
  10.     // 獲取所有的 請求服務 
  11.     const promiseApi = [] 
  12.     apis.forEach(api => { 
  13.         // 利用promise  
  14.         let promise = promiseCache.get(api) 
  15.         if (promise) { 
  16.             // 如果 緩存中有,直接push 
  17.             promise.push(promise) 
  18.         } else { 
  19.              promise = request.get(querys[api]).then(res => { 
  20.                 // 對res 進行操作 
  21.                 ... 
  22.                 }).catch(error => { 
  23.                 // 在請求回來后,如果出現(xiàn)問題,把promise從cache中刪除 
  24.                 promiseCache.delete(api) 
  25.                 return Promise.reject(error) 
  26.             }) 
  27.             promiseCache.set(api, promise) 
  28.             promiseCache.push(promise) 
  29.         } 
  30.     }) 
  31.     return Promise.all(promiseApi).then(res => { 
  32.         // 根據(jù)傳入的 是字符串還是數(shù)組來返回數(shù)據(jù),因為本身都是數(shù)組操作 
  33.         // 如果傳入的是字符串,則需要取出操作 
  34.         return queryIsArray ? res : res[0] 
  35.     }) 
  36. }

該方案是同時獲取多個服務器數(shù)據(jù)的方式。可以同時獲得多個數(shù)據(jù)進行操作,不會因為單個數(shù)據(jù)出現(xiàn)問題而發(fā)生錯誤。

調用方式:

 
 
 
  1. queryAll('wares').then( ... ) 
  2. // 第二次調用 不會去取 wares,只會去skus 
  3. queryAll(['wares', 'skus']).then( ... )

方案四 、添加時間有關的緩存

往往緩存是有危害的,如果我們在知道修改了數(shù)據(jù)的情況下,直接把 cache 刪除即可,此時我們調用方法就可以向服務器進行請求。

這樣我們規(guī)避了前端顯示舊的的數(shù)據(jù)。但是我們可能一段時間沒有對數(shù)據(jù)進行操作,那么此時舊的數(shù)據(jù)就一直存在,那么我們最好規(guī)定個時間來去除數(shù)據(jù)。

該方案是采用了 類 持久化數(shù)據(jù)來做數(shù)據(jù)緩存,同時添加了過期時長數(shù)據(jù)以及參數(shù)化。

代碼如下:

首先定義持久化類,該類可以存儲 promise 或者 data

 
 
 
  1. class ItemCache() { 
  2.     construct(data, timeout) { 
  3.         this.data = data 
  4.         // 設定超時時間,設定為多少秒 
  5.         this.timeout = timeout 
  6.         // 創(chuàng)建對象時候的時間,大約設定為數(shù)據(jù)獲得的時間 
  7.         this.cacheTime = (new Date()).getTime 
  8.     } 
  9. }

然后我們定義該數(shù)據(jù)緩存。我們采用Map 基本相同的api

 
 
 
  1. class ExpriesCache { 
  2.     // 定義靜態(tài)數(shù)據(jù)map來作為緩存池 
  3.     static cacheMap =  new Map() 
  4.     // 數(shù)據(jù)是否超時 
  5.     static isOverTime(name) { 
  6.         const data = ExpriesCache.cacheMap.get(name) 
  7.         // 沒有數(shù)據(jù) 一定超時 
  8.         if (!data) return true 
  9.         // 獲取系統(tǒng)當前時間戳 
  10.         const currentTime = (new Date()).getTime()   
  11.           // 獲取當前時間與存儲時間的過去的秒數(shù) 
  12.         const overTime = (currentTime - data.cacheTime) / 1000 
  13.         // 如果過去的秒數(shù)大于當前的超時時間,也返回null讓其去服務端取數(shù)據(jù) 
  14.         if (Math.abs(overTime) > data.timeout) { 
  15.             // 此代碼可以沒有,不會出現(xiàn)問題,但是如果有此代碼,再次進入該方法就可以減少判斷。 
  16.             ExpriesCache.cacheMap.delete(name) 
  17.             return true 
  18.         } 
  19.         // 不超時 
  20.         return false
  21.     } 
  22.     // 當前data在 cache 中是否超時 
  23.     static has(name) { 
  24.         return !ExpriesCache.isOverTime(name) 
  25.     } 
  26.     // 刪除 cache 中的 data 
  27.     static delete(name) { 
  28.         return ExpriesCache.cacheMap.delete(name) 
  29.     } 
  30.     // 獲取 
  31.     static get(name) { 
  32.         const isDataOverTiem = ExpriesCache.isOverTime(name) 
  33.         //如果 數(shù)據(jù)超時,返回null,但是沒有超時,返回數(shù)據(jù),而不是 ItemCache 對象 
  34.         return isDataOverTiem ? null : ExpriesCache.cacheMap.get(name).data 
  35.     } 
  36.     // 默認存儲20分鐘 
  37.     static set(name, data, timeout = 1200) { 
  38.         // 設置 itemCache 
  39.         const itemCache = mew ItemCache(data, timeout) 
  40.         //緩存 
  41.         ExpriesCache.cacheMap.set(name, itemCache) 
  42.     } 
  43. }

此時數(shù)據(jù)類以及操作類 都已經定義好,我們可以在api層這樣定義

 
 
 
  1. // 生成key值錯誤 
  2. const generateKeyError = new Error("Can't generate key from name and argument") 
  3. // 生成key值 
  4. function generateKey(name, argument) { 
  5.     // 從arguments 中取得數(shù)據(jù)然后變?yōu)閿?shù)組 
  6.     const params = Array.from(argument).join(',') 
  7.     try{ 
  8.         // 返回 字符串,函數(shù)名 + 函數(shù)參數(shù) 
  9.         return `${name}:${params}` 
  10.     }catch(_) { 
  11.         // 返回生成key錯誤 
  12.         return generateKeyError 
  13.     } 
  14. async getWare(params1, params2) { 
  15.     // 生成key 
  16.     const key = generateKey('getWare', [params1, params2])  
  17.     // 獲得數(shù)據(jù) 
  18.     let data = ExpriesCache.get(key) 
  19.     if (!data) { 
  20.         const res = await request('/getWares', {params1, params2}) 
  21.         // 使用 10s 緩存,10s之后再次get就會 獲取null 而從服務端繼續(xù)請求 
  22.         ExpriesCache.set(key, res, 10) 
  23.     } 
  24.     return data 
  25. }

該方案使用了 過期時間 和 api 參數(shù)不同而進行 緩存的方式。已經可以滿足絕大部分的業(yè)務場景。

調用方式:

 
 
 
  1. getWares(1,2).then( ... ) 
  2. // 第二次調用 取得先前的promise 
  3. getWares(1,2).then( ... ) 
  4. // 不同的參數(shù),不取先前promise 
  5. getWares(1,3).then( ... )

方案五、基于修飾器的方案四

和方案四是的解法一致的,但是是基于修飾器來做。

代碼如下:

 
 
 
  1. // 生成key值錯誤 
  2. const generateKeyError = new Error("Can't generate key from name and argument") 
  3. // 生成key值 
  4. function generateKey(name, argument) { 
  5.     // 從arguments 中取得數(shù)據(jù)然后變?yōu)閿?shù)組 
  6.     const params = Array.from(argument).join(',') 
  7.     try{ 
  8.         // 返回 字符串 
  9.         return `${name}:${params}` 
  10.     }catch(_) { 
  11.         return generateKeyError 
  12.     } 
  13. function decorate(handleDescription, entryArgs) { 
  14.     // 判斷 當前 最后數(shù)據(jù)是否是descriptor,如果是descriptor,直接 使用 
  15.     // 例如 log 這樣的修飾器 
  16.     if (isDescriptor(entryArgs[entryArgs.length - 1])) { 
  17.         return handleDescription(...entryArgs, []) 
  18.     } else { 
  19.         // 如果不是 
  20.         // 例如 add(1) plus(20) 這樣的修飾器 
  21.         return function() { 
  22.             return handleDescription(...Array.protptype.slice.call(arguments), entryArgs) 
  23.         } 
  24.     } 
  25. function handleApiCache(target, name, descriptor, ...config) { 
  26.     // 拿到函數(shù)體并保存 
  27.     const fn = descriptor.value 
  28.     // 修改函數(shù)體 
  29.     descriptor.value = function () {  
  30.         const key =  generateKey(name, arguments) 
  31.         // key無法生成,直接請求 服務端數(shù)據(jù) 
  32.         if (key === generateKeyError)  { 
  33.             // 利用剛才保存的函數(shù)體進行請求 
  34.             return fn.apply(null, arguments) 
  35.         } 
  36.         let promise = ExpriesCache.get(key) 
  37.         if (!promise) { 
  38.             // 設定promise 
  39.             promise = fn.apply(null, arguments).catch(error => { 
  40.                  // 在請求回來后,如果出現(xiàn)問題,把promise從cache中刪除 
  41.                 ExpriesCache.delete(key) 
  42.                 // 返回錯誤 
  43.                 return Promise.reject(error) 
  44.             }) 
  45.             // 使用 10s 緩存,10s之后再次get就會 獲取null 而從服務端繼續(xù)請求 
  46.             ExpriesCache.set(key, promise, config[0]) 
  47.         } 
  48.         return promise  
  49.     } 
  50.     return descriptor; 
  51. }
  52. // 制定 修飾器 
  53. function ApiCache(...args) { 
  54.     return decorate(handleApiCache, args) 
  55. }

此時 我們就會使用 類來對api進行緩存

 
 
 
  1. class Api { 
  2.     // 緩存10s 
  3.     @ApiCache(10) 
  4.     // 此時不要使用默認值,因為當前 修飾器 取不到 
  5.     getWare(params1, params2) { 
  6.         return request.get('/getWares') 
  7.     } 
  8. }

因為函數(shù)存在函數(shù)提升,所以沒有辦法利用函數(shù)來做 修飾器

例如:

 
 
 
  1. var counter = 0; 
  2. var add = function () { 
  3.   counter++; 
  4. }; 
  5. @add 
  6. function foo() { 
  7. }

該代碼意圖是執(zhí)行后counter等于 1,但是實際上結果是counter等于 0。因為函數(shù)提升,使得實際執(zhí)行的代碼是下面這樣

 
 
 
  1. @add 
  2. function foo() { 
  3. var counter;
  4. var add; 
  5. counter = 0; 
  6. add = function () { 
  7.   counter++; 
  8. };

所以沒有 辦法在函數(shù)上用修飾器。具體參考ECMAScript 6 入門 Decorator

此方式寫法簡單且對業(yè)務層沒有太多影響。但是不可以動態(tài)修改 緩存時間

調用方式

 
 
 
  1. getWares(1,2).then( ... ) 
  2. // 第二次調用 取得先前的promise 
  3. getWares(1,2).then( ... ) 
  4. // 不同的參數(shù),不取先前promise 
  5. getWares(1,3).then( ... )

總結

api的緩存機制與場景在這里也基本上介紹了,基本上能夠完成絕大多數(shù)的數(shù)據(jù)業(yè)務緩存,在這里我也想請教教大家,有沒有什么更好的解決方案,或者這篇博客中有什么不對的地方,歡迎指正,在這里感謝各位了。

同時這里也有很多沒有做完的工作,可能會在后面的博客中繼續(xù)完善。


網站標題:前端api請求緩存方案
轉載注明:http://m.5511xx.com/article/codihoe.html