摘要:但是這篇文章除去科普基本概念外,更重要的就是介紹中間件思想,并自己來(lái)實(shí)現(xiàn)一個(gè)服務(wù)端緩存中間件。
Express 作為 Node.js 的框架,如今發(fā)展可謂如日中天。我很喜歡其靈活、易擴(kuò)展的設(shè)計(jì)理念。尤其是該框架的中間件架構(gòu)設(shè)計(jì):使得在應(yīng)用中加入新特性更加標(biāo)準(zhǔn)化、成本最小化。這篇文章,我會(huì)嘗試編寫(xiě)一個(gè)非常簡(jiǎn)單、小巧的中間件,完成服務(wù)端緩存功能,進(jìn)而優(yōu)化性能。
關(guān)于中間件說(shuō)到中間件,Express 官網(wǎng)對(duì)它的闡述是這樣的:
“Express 是一個(gè)自身功能極簡(jiǎn),完全是路由和中間件構(gòu)成一個(gè)web開(kāi)發(fā)框架:從本質(zhì)上來(lái)說(shuō),一個(gè) Express 應(yīng)用就是在調(diào)用各種中間件。”
也許你使用過(guò)各種各樣的中間件進(jìn)行開(kāi)發(fā),但是可能并不理解中間件原理,也沒(méi)有深入過(guò) Express 源碼,探究其實(shí)現(xiàn)。這里并不打算長(zhǎng)篇大論幫您分析,但是使用層面上大致可以參考下圖:
建議有興趣、想深入的讀者自己分析,有任何問(wèn)題歡迎與我討論。即便您不打算深入,也不會(huì)影響對(duì)下文中間件編寫(xiě)的理解。
關(guān)于服務(wù)端緩存緩存已經(jīng)被廣泛應(yīng)用,來(lái)提高頁(yè)面性能。一說(shuō)到緩存,可能讀者腦海里馬上冒出來(lái):“客戶端緩存,CDN 緩存,服務(wù)器端緩存......”。另一維度上,也會(huì)想到:“200(from cache),expire,eTag......”等概念。
當(dāng)然作為前端開(kāi)發(fā)者,我們一定要明白這些緩存概念,這些緩存理念是相對(duì)于某個(gè)具體用戶訪問(wèn)來(lái)說(shuō)的,性能優(yōu)化體現(xiàn)在單個(gè)用戶上。比如說(shuō),我第一次打開(kāi)頁(yè)面 A,耗時(shí)超長(zhǎng),下一次打開(kāi)頁(yè)面由于緩存的作用,時(shí)間縮短了。
但是在服務(wù)器端,還存在另外一個(gè)維度,思考一下這樣的場(chǎng)景:
我們有一個(gè)靜態(tài)頁(yè)面 B,這個(gè)頁(yè)面服務(wù)端需要從數(shù)據(jù)庫(kù)獲取部分?jǐn)?shù)據(jù) b1,根據(jù) b1 又要計(jì)算得到部分?jǐn)?shù)據(jù) b2,還得做各種高復(fù)雜度操作,最終才能“東拼西湊”出需要返回的完整頁(yè)面 B,整個(gè)過(guò)程耗時(shí)2s。
那么面臨的災(zāi)難就是,user1 打開(kāi)頁(yè)面耗時(shí)2s,user2同樣打開(kāi)頁(yè)面耗時(shí)2s......而這些頁(yè)面都是靜態(tài)頁(yè)面 B,內(nèi)容是完全一樣的。為了解決這個(gè)災(zāi)難,這時(shí)候我們也需要緩存,這種緩存就叫先做服務(wù)端緩存(server-side cache)。
總結(jié)一下,服務(wù)端緩存的目的其實(shí)就是對(duì)于同一個(gè)頁(yè)面請(qǐng)求,而返回(緩存的)同樣的頁(yè)面內(nèi)容。這個(gè)過(guò)程完全獨(dú)立于不同的用戶。
上面的話有些拗口,可以參考英文表達(dá)更清晰:
The goal of server side cache is responding to the same content for the same request independently of the client’s request.
因此,下面展示的 demo 在第一次請(qǐng)求到達(dá)時(shí),服務(wù)端耗費(fèi)5秒來(lái)返回 HTML;而接下來(lái)再次請(qǐng)求該頁(yè)面,將會(huì)命中緩存,不過(guò)是哪個(gè)用戶訪問(wèn),只需要幾毫秒便可得到完整頁(yè)面。
Show me the code & Demo其實(shí)上文提到的緩存概念非常簡(jiǎn)單,稍微有些后端經(jīng)驗(yàn)的同學(xué)都能很好理解。但是這篇文章除去科普基本概念外,更重要的就是介紹 Express 中間件思想,并自己來(lái)實(shí)現(xiàn)一個(gè)服務(wù)端緩存中間件。
讓我們開(kāi)工吧!
最終 Demo 代碼,歡迎訪問(wèn)它的Github地址。
我將會(huì)使用 npm 上 memory-cache 這個(gè)包,以方便進(jìn)行緩存的讀寫(xiě)。最終的中間件代碼很簡(jiǎn)單:
"use strict" var mcache = require("memory-cache"); var cache = (duration) => { return (req, res, next) => { let key = "__express__" + req.originalUrl || req.url let cachedBody = mcache.get(key) if (cachedBody) { res.send(cachedBody) return } else { res.sendResponse = res.send res.send = (body) => { mcache.put(key, body, duration * 1000); res.sendResponse(body) } next() } } }
為了簡(jiǎn)單,我使用了請(qǐng)求 URL 作為 cache 的 key:
當(dāng)它(cache key)及其對(duì)應(yīng)的 value 值存在時(shí),便直接返回其 value 值;
當(dāng)它(cache key)及其對(duì)應(yīng)的 value 值不存在時(shí),我們將對(duì) Express send 方法做一層攔截:在最終返回前,存入這對(duì) key-value。
緩存的有效時(shí)間是10秒。
最終在判斷之外,我們的中間件把控制權(quán)交給下一個(gè)中間件。
最終使用和測(cè)試如下代碼:
app.get("/", cache(10), (req, res) => { setTimeout(() => { res.render("index", { title: "Hey", message: "Hello there", date: new Date()}) }, 5000) //setTimeout was used to simulate a slow processing request })
我使用了 setTimeout 來(lái)模擬一個(gè)超長(zhǎng)(5s)的操作。
打開(kāi)瀏覽器控制面板,發(fā)現(xiàn)在10秒緩存到期以內(nèi):
至于為什么 cache 中間件要那樣子寫(xiě)、next() 為什么是中間件把控制權(quán)傳遞,我并不打算展開(kāi)去講。有興趣的讀者可以看一下 Express 源碼。
還有幾個(gè)小問(wèn)題仔細(xì)看我們的頁(yè)面,再去體會(huì)一下實(shí)現(xiàn)代碼。也許細(xì)心的讀者能發(fā)現(xiàn)一個(gè)問(wèn)題:剛才的實(shí)現(xiàn)我們緩存了整個(gè)頁(yè)面,并將 date: new Date() 傳入了 jade 模版 index.jade 里。那么,在命中緩存的條件下,10秒內(nèi),頁(yè)面無(wú)法動(dòng)態(tài)刷新來(lái)同步,直到10秒緩存到期。
同時(shí),我們什么時(shí)候可以使用上述中間件,進(jìn)行服務(wù)端緩存呢?當(dāng)然是靜態(tài)內(nèi)容才可以使用。同時(shí),PUT, DELETE 和 POST 操作都不應(yīng)該進(jìn)行類似的緩存處理。
同樣,我們使用了 npm 模塊:memory-cache,它存在優(yōu)缺點(diǎn)如下:
讀寫(xiě)迅速而簡(jiǎn)單,不需要其他依賴;
當(dāng)服務(wù)器或者這個(gè)進(jìn)程掛掉的時(shí)候,緩存中的內(nèi)容將會(huì)全部丟失。
memcache 是將緩存內(nèi)容存放在了自己進(jìn)程的內(nèi)存中,所以這部分內(nèi)容是無(wú)法在多個(gè) Node.js 進(jìn)程之間共享的。
如果這些弊端 really matter,在實(shí)際開(kāi)發(fā)中我們可以選擇分布式的 cache 服務(wù),比如 Redis。同樣你可以在 npm 上找到:express-redis-cache 模塊使用。
總結(jié)在真實(shí)的開(kāi)發(fā)場(chǎng)景中,服務(wù)端緩存已經(jīng)成為 common sense,但是在 Node.js 的世界里,體會(huì)其中間件思想,自己手動(dòng)編寫(xiě)服務(wù),同樣樂(lè)趣無(wú)窮。
與實(shí)踐相結(jié)合,我認(rèn)為真正緩存整個(gè)頁(yè)面(如同 demo 那樣)并不是一個(gè)推薦的做法(當(dāng)時(shí)實(shí)際場(chǎng)景實(shí)際分析),同樣使用請(qǐng)求 url 作為緩存的 key 也有待考慮。比如,頁(yè)面中的一些靜態(tài)內(nèi)容可能會(huì)在其他頁(yè)面中重復(fù)使用到,復(fù)用就成了問(wèn)題。
真實(shí)場(chǎng)景下,一切設(shè)計(jì)和邏輯都要為自己業(yè)務(wù)情況所負(fù)責(zé)。脫離需求談實(shí)現(xiàn),都是耍流氓。這個(gè) demo 簡(jiǎn)易輕巧,有需要的讀者可以訪問(wèn)它的Github地址,歡迎玩出各種花樣。
Happy Coding!
PS:
作者Github倉(cāng)庫(kù) 和 知乎問(wèn)答鏈接
歡迎各種形式交流。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.hztianpu.com/yun/88400.html
摘要:前端日?qǐng)?bào)精選機(jī)制詳解與中實(shí)踐應(yīng)用基礎(chǔ)與實(shí)踐如何用獲取虛擬鍵盤(pán)高度適用所有平臺(tái)和入門(mén)教程阮一峰的網(wǎng)絡(luò)日志編程技能提升指南中文到底什么是又是什么眾成翻譯調(diào)用模塊騰訊前端團(tuán)隊(duì)社區(qū)小書(shū)從一個(gè)簡(jiǎn)單的例子講起小書(shū)教程小書(shū)優(yōu)化操作小書(shū)教 2017-09-07 前端日?qǐng)?bào) 精選 JavaScript Event Loop 機(jī)制詳解與 Vue.js 中實(shí)踐應(yīng)用 Redux 基礎(chǔ)與實(shí)踐如何用 js 獲取虛擬...
摘要:這個(gè)由提供的工具。在轉(zhuǎn)換步驟是異步時(shí),你可以這樣告訴本次轉(zhuǎn)換是異步的,會(huì)在中回調(diào)結(jié)果通過(guò)返回異步執(zhí)行后的結(jié)果參考編寫(xiě)一個(gè) 本文示例源代碼請(qǐng)戳github博客,建議大家動(dòng)手敲敲代碼。 本文不會(huì)介紹loader的一些使用方法,不熟悉的同學(xué)請(qǐng)自行查看Webpack loader 1、背景 首先我們來(lái)看一下為什么需要loader,以及他能干什么?webpack 只能理解 JavaScript...
摘要:前言一直很喜歡看科技新聞,多年來(lái)一直混跡于,以前西貝的評(píng)論區(qū)是匿名的,所以評(píng)論區(qū)非?;钴S,各種噴子和段子,不過(guò)也確實(shí)很歡樂(lè),可以說(shuō)那是西貝人氣最旺的時(shí)候。 前言 一直很喜歡看科技新聞,多年來(lái)一直混跡于cnBeta,以前西貝的評(píng)論區(qū)是匿名的,所以評(píng)論區(qū)非常活躍,各種噴子和段子,不過(guò)也確實(shí)很歡樂(lè),可以說(shuō)那是西貝人氣最旺的時(shí)候。然而自從去年網(wǎng)信辦出臺(tái)了《互聯(lián)網(wǎng)跟帖評(píng)論服務(wù)管理規(guī)定》,要求只有...
摘要:前言一直很喜歡看科技新聞,多年來(lái)一直混跡于,以前西貝的評(píng)論區(qū)是匿名的,所以評(píng)論區(qū)非?;钴S,各種噴子和段子,不過(guò)也確實(shí)很歡樂(lè),可以說(shuō)那是西貝人氣最旺的時(shí)候。 前言 一直很喜歡看科技新聞,多年來(lái)一直混跡于cnBeta,以前西貝的評(píng)論區(qū)是匿名的,所以評(píng)論區(qū)非?;钴S,各種噴子和段子,不過(guò)也確實(shí)很歡樂(lè),可以說(shuō)那是西貝人氣最旺的時(shí)候。然而自從去年網(wǎng)信辦出臺(tái)了《互聯(lián)網(wǎng)跟帖評(píng)論服務(wù)管理規(guī)定》,要求只有...
閱讀 2604·2021-09-29 09:34
閱讀 3417·2021-09-23 11:21
閱讀 2594·2021-09-06 15:00
閱讀 1183·2019-08-30 15:44
閱讀 2092·2019-08-29 17:23
閱讀 3084·2019-08-29 16:44
閱讀 3122·2019-08-29 13:13
閱讀 2035·2019-08-28 18:12