0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

從2s優(yōu)化到0.1s,這周班沒白加!

CodeSheep ? 來源:CodeSheep ? 2023-06-28 17:03 ? 次閱讀

大家好,今天我們繼續(xù)來分享一個在項目開發(fā)過程中遇到的實際問題,這里也來梳理并總結(jié)一下我們是如何對它進行持續(xù)優(yōu)化的,希望能對大家有所幫助。

分類樹查詢功能,在各個業(yè)務(wù)系統(tǒng)中可以說隨處可見,特別是在一些電商系統(tǒng)中。

b2a0caea-1588-11ee-962d-dac502259ad0.png

但就是這樣一個看似簡單的分類樹查詢功能,我們卻優(yōu)化了數(shù)次。

這其中到底經(jīng)歷了什么呢?

背 景

我們的一個老項目使用了SpringBoot推薦的模板引擎:Thymeleaf,進行動態(tài)渲染。

它是一個XML/XHTML/HTML5模板引擎,可用于Web與非Web環(huán)境中的應(yīng)用開發(fā)。

它提供了一個用于整合SpringMVC的可選模塊,在應(yīng)用開發(fā)中,我們可以使用Thymeleaf來完全代替JSP或其他模板引擎,如VelocityFreeMarker等。

前端開發(fā)寫好Thymeleaf的模板文件,調(diào)用后端接口獲取數(shù)據(jù),進行動態(tài)綁定,就能把想要的內(nèi)容展示給用戶。

由于當(dāng)時很早這也個是從0-1的新項目,為了開快速開發(fā)功能,第一版接口是直接從數(shù)據(jù)庫中查詢分類數(shù)據(jù),組裝成分類樹,然后返回給前端。

通過這種方式,簡化了數(shù)據(jù)流程,快速把整個頁面功能調(diào)通了。

第1次優(yōu)化

我們將該接口部署到dev環(huán)境,剛開始沒啥問題。

隨著開發(fā)人員添加的分類越來越多,很快就暴露出性能瓶頸。

我們不得不做優(yōu)化了。

我們第一個想到的是:加Redis緩存

流程圖如下:

b2c6e8ce-1588-11ee-962d-dac502259ad0.png

于是暫時這樣優(yōu)化了一下:

  1. 用戶訪問接口獲取分類樹時,先從Redis中查詢數(shù)據(jù)。
  2. 如果Redis中有數(shù)據(jù),則直接數(shù)據(jù)。
  3. 如果Redis中沒有數(shù)據(jù),則再從數(shù)據(jù)庫中查詢數(shù)據(jù),拼接成分類樹返回。
  4. 將從數(shù)據(jù)庫中查到的分類樹的數(shù)據(jù),保存到Redis中,設(shè)置過期時間5分鐘。
  5. 將分類樹返回給用戶。

我們在Redis中定義一個了key,value是一個分類樹的json格式轉(zhuǎn)換成了字符串,使用簡單的key/value形式保存數(shù)據(jù)。

經(jīng)過這樣優(yōu)化之后,dev環(huán)境的聯(lián)調(diào)和自測順利完成了。

第2次優(yōu)化

我們將這個功能部署到st環(huán)境了。

剛開始測試同學(xué)沒有發(fā)現(xiàn)什么問題,但隨著后面不斷地深入測試,隔一段時間就出現(xiàn)一次首頁訪問很慢的情況。

于是,我們馬上進行了第2次優(yōu)化。

我們決定使用Job定期異步更新分類樹到Redis中,在系統(tǒng)上線之前,會先生成一份數(shù)據(jù)。

當(dāng)然為了保險起見,防止Redis在哪條突然掛了,之前分類樹同步寫入Redis的邏輯還是保留。

于是,流程圖改成了這樣:

b2f15a96-1588-11ee-962d-dac502259ad0.png

增加了一個job每隔5分鐘執(zhí)行一次,從數(shù)據(jù)庫中查詢分類數(shù)據(jù),封裝成分類樹,更新到Redis緩存中。

其他的流程保持不變。

此外,Redis的過期時間之前設(shè)置的5分鐘,現(xiàn)在要改成永久。

通過這次優(yōu)化之后,st環(huán)境就沒有再出現(xiàn)過分類樹查詢的性能問題了。

第3次優(yōu)化

測試了一段時間之后,整個網(wǎng)站的功能快要上線了。

為了保險起見,我們需要對網(wǎng)站首頁做一次壓力測試。

果然測出問題了,網(wǎng)站首頁最大的qps是100多,最后發(fā)現(xiàn)是每次都從Redis獲取分類樹導(dǎo)致的網(wǎng)站首頁的性能瓶頸。

我們需要做第3次優(yōu)化。

該怎么優(yōu)化呢?

答:加內(nèi)存緩存。

如果加了內(nèi)存緩存,就需要考慮數(shù)據(jù)一致性問題。

內(nèi)存緩存是保存在服務(wù)器節(jié)點上的,不同的服務(wù)器節(jié)點更新的頻率可能有點差異,這樣可能會導(dǎo)致數(shù)據(jù)的不一致性。

但分類本身是更新頻率比較低的數(shù)據(jù),對于用戶來說不太敏感,即使在短時間內(nèi),用戶看到的分類樹有些差異,也不會對用戶造成太大的影響。

因此,分類樹這種業(yè)務(wù)場景,是可以使用內(nèi)存緩存的。

于是,我們使用了Spring推薦的caffine作為內(nèi)存緩存。

改造后的流程圖如下:

b320f4b8-1588-11ee-962d-dac502259ad0.png

  1. 用戶訪問接口時改成先從本地緩存分類數(shù)查詢數(shù)據(jù)。
  2. 如果本地緩存有,則直接返回。
  3. 如果本地緩存沒有,則從Redis中查詢數(shù)據(jù)。
  4. 如果Redis中有數(shù)據(jù),則將數(shù)據(jù)更新到本地緩存中,然后返回數(shù)據(jù)。
  5. 如果Redis中也沒有數(shù)據(jù)(說明Redis掛了),則從數(shù)據(jù)庫中查詢數(shù)據(jù),更新到Redis中(萬一Redis恢復(fù)了呢),然后更新到本地緩存中,返回返回數(shù)據(jù)。

需要注意的是,需要改本地緩存設(shè)置一個過期時間,這里設(shè)置的5分鐘,不然的話,沒辦法獲取新的數(shù)據(jù)。

這樣優(yōu)化之后,再次做網(wǎng)站首頁的壓力測試,qps提升到了500多,滿足上線要求。

第4次優(yōu)化

之后,這個功能順利上線了。

使用了很長一段時間沒有出現(xiàn)問題。

兩年后的某一天,有用戶反饋說,網(wǎng)站首頁有點慢。

我們排查了一下原因發(fā)現(xiàn),分類樹的數(shù)據(jù)太多了,一次性返回了上萬個分類。

原來在系統(tǒng)上線的這兩年多的時間內(nèi),運營同學(xué)在系統(tǒng)后臺增加了很多分類。

我們需要做第4次優(yōu)化。

這時要如何優(yōu)化呢?

限制分類樹的數(shù)量?

答:也不太現(xiàn)實,目前這個業(yè)務(wù)場景就是有這么多分類,不能讓用戶選擇不到他想要的分類吧?

這時我們想到最快的辦法是開啟nginxGZip功能。

讓數(shù)據(jù)在傳輸之前,先壓縮一下,然后進行傳輸,在用戶瀏覽器中,自動解壓,將真實的分類樹數(shù)據(jù)展示給用戶。

之前調(diào)用接口返回的分類樹有1MB的大小,優(yōu)化之后,接口返回的分類樹的大小是100Kb,一下子縮小了10倍。

這樣簡單的優(yōu)化之后,性能提升了一些。

第5次優(yōu)化

經(jīng)過上面優(yōu)化之后,用戶很長一段時間都沒有反饋性能問題。

但有一天公司同事在排查Redis中大key的時候,揪出了分類樹。之前的分類樹使用key/value的結(jié)構(gòu)保存數(shù)據(jù)的。

我們不得不做第5次優(yōu)化。

為了優(yōu)化在Redis中存儲數(shù)據(jù)的大小,我們首先需要對數(shù)據(jù)進行瘦身。

只保存需要用到的字段。

例如:

@AllArgsConstructor
@Data
publicclassCategory{

privateLongid;
privateStringname;
privateLongparentId;
privateDateinDate;
privateLonginUserId;
privateStringinUserName;
privateListchildren;
}

像這個分類對象中inDate、inUserId和inUserName字段是可以不用保存的。

修改自動名稱。

例如:

@AllArgsConstructor
@Data
publicclassCategory{
/**
*分類編號
*/
@JsonProperty("i")
privateLongid;

/**
*分類層級
*/
@JsonProperty("l")
privateIntegerlevel;

/**
*分類名稱
*/
@JsonProperty("n")
privateStringname;

/**
*父分類編號
*/
@JsonProperty("p")
privateLongparentId;

/**
*子分類列表
*/
@JsonProperty("c")
privateListchildren;
}

由于在一萬多條數(shù)據(jù)中,每條數(shù)據(jù)的字段名稱是固定的,他們的重復(fù)率太高了。

由此,可以在json序列化時,改成一個簡短的名稱,以便于返回更少的數(shù)據(jù)大小。

這還不夠,需要對存儲的數(shù)據(jù)做壓縮。

之前在Redis中保存的key/value,其中的value是json格式的字符串。

其實RedisTemplate支持,value保存byte數(shù)組。

先將json字符串?dāng)?shù)據(jù)用GZip工具類壓縮成byte數(shù)組,然后保存到Redis中。

再獲取數(shù)據(jù)時,將byte數(shù)組轉(zhuǎn)換成json字符串,然后再轉(zhuǎn)換成分類樹。

這樣優(yōu)化之后,保存到Redis中的分類樹的數(shù)據(jù)大小,一下子減少了10倍,Redis的大key問題被解決了。

小 結(jié)

所以回過頭來看,這樣一個看似并不復(fù)雜的功能需求,但是要想把它做到穩(wěn)定、高效、可用,一路下來還是需要考慮不少問題的。而這其中遇到的任何一個問題,一旦解決并復(fù)盤了,它也就匯聚成我們的經(jīng)驗了,希望這篇文章的梳理能對大家有所幫助。

好了,今天的內(nèi)容分享就到這里了,感謝大家的收看,我們下篇見。



聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 數(shù)據(jù)庫
    +關(guān)注

    關(guān)注

    7

    文章

    3739

    瀏覽量

    64173
  • 模板
    +關(guān)注

    關(guān)注

    0

    文章

    108

    瀏覽量

    20542
  • 分類樹
    +關(guān)注

    關(guān)注

    0

    文章

    3

    瀏覽量

    5736

原文標(biāo)題:從2s優(yōu)化到0.1s,這周班沒白加!

文章出處:【微信號:CodeSheep,微信公眾號:CodeSheep】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    S2S(Services To Signal)測試問答#S2S測試 #SOA

    S2S
    北匯信息POLELINK
    發(fā)布于 :2023年01月19日 17:37:52

    請問按鍵2S如何實現(xiàn)啊

    請問按鍵2S如何實現(xiàn)啊我現(xiàn)在用的定時器時間為25MS如何實現(xiàn)按鍵2S請高手給予指點 最好是簡單的代碼
    發(fā)表于 05-10 11:31

    0.1S~6小時定時電路圖

    0.1S~6小時定時電路圖:本定時器電路由基準(zhǔn)脈沖產(chǎn)生器,分頻電路和執(zhí)行電路組成,分單步定時和循環(huán)定時。
    發(fā)表于 10-27 09:28 ?1135次閱讀
    <b class='flag-5'>0.1S</b>~6小時定時電路圖

    全“芯”升級 小米手機2S詳細拆解

    小米手機在國產(chǎn)智能手機中一直是人們關(guān)注的焦點,除了1999元經(jīng)典的平民價格外,其旗艦機水準(zhǔn)的配置總能在第一時間抓住人眼球。小米手機2S 是小米手機2的升級版本,就外觀而言,消費者根本無從分辨二者
    的頭像 發(fā)表于 04-14 15:45 ?3.5w次閱讀

    好高興又可以用小米2S啦!你的小米2S還在嗎?

    小米2S可以說是小米系列手機里最經(jīng)典的手機,沒有之一。如今小米更迭很快,依舊為發(fā)燒而生,可是卻再也沒有一款像當(dāng)年小米2S那樣經(jīng)典之作
    發(fā)表于 03-03 08:16 ?4935次閱讀
    好高興又可以用小米<b class='flag-5'>2S</b>啦!你的小米<b class='flag-5'>2S</b>還在嗎?

    華為nova 2s四鏡頭?功能的升級可能比你想象的要多

    今日,華為官方在社交平臺上發(fā)布的新機華為nova 2s的H5邀請函,引起了媒體和網(wǎng)友的廣泛關(guān)注,在H5內(nèi)容中,流量小生張藝興間接宣布繼續(xù)擔(dān)任nova 2s代言人,華為nova 2s海報也作為彩蛋在
    發(fā)表于 12-03 12:09 ?2441次閱讀

    s=surprise,發(fā)布會后大家都這樣評價nova 2s

    12月7日下午,華為正式發(fā)布了華為 nova 2s,作為nova系列的最新一代產(chǎn)品,華為 nova 2s繼承了高顏值的外觀設(shè)計以及強大的自拍、拍照性能,并且加入了很多最新的時尚元素,比如說采用全面屏
    發(fā)表于 12-08 16:17 ?1079次閱讀

    來了來了!最新小米MIX 2S曝光

    近日,小米發(fā)布預(yù)告稱將于3月27日發(fā)布小米MIX 2S,目前有爆料為我們揭露了這款新機的謎團。
    的頭像 發(fā)表于 03-02 14:16 ?3100次閱讀

    小米MIX 2S評測:全面屏好看,但驚喜難以勾起購買的欲望

    小米MIX 2S在配置方面,依然是走了性能路線,擁有“高通845+8GB”這目前國產(chǎn)機當(dāng)中頂尖的配置。然而這次小米MIX 2S的出現(xiàn),并沒有當(dāng)初小米MIX出現(xiàn)時候的那種驚艷,外觀設(shè)計沒有大的進步,而背面類似于iPhone X的設(shè)計在大眾評價當(dāng)中,甚至有退步的感覺。
    的頭像 發(fā)表于 04-05 15:00 ?5316次閱讀

    小米 MIX 2S 與 MIX 2 對比圖賞

    小米在 3 月底給我們帶來了全新的旗艦——MIX 2S,延續(xù)了 MIX 系列的全面屏設(shè)計,這一次除完成國內(nèi)廠商首發(fā)驍龍 845 的任務(wù)外,更是大幅度提升了以往 MIX 系列的拍照弱項。在詳細評測出來之前,我們先來看看小米 MIX 2S 與 MIX
    的頭像 發(fā)表于 04-13 10:19 ?9080次閱讀

    谷歌ARCore登陸中國市場 小米Mix 2S應(yīng)用商店首發(fā)

    在小米8發(fā)布之前,小米MIX 2S可以說是小米目前旗艦手機的最佳代表,這款產(chǎn)品發(fā)布于3月27日,憑借著小米史上最好拍照手機的頭銜,MIX 2S一經(jīng)發(fā)售就受到了用戶的追捧,不過由于剛發(fā)售時候的產(chǎn)能原因,小米MIX 2S經(jīng)常出現(xiàn)斷貨
    發(fā)表于 05-30 09:56 ?1244次閱讀

    小米Air 2s無線耳機固件更新

    。 本次固件更新優(yōu)化彈窗問題;優(yōu)化聲音卡頓、音質(zhì)、LHDC;優(yōu)化誤觸、提示燈等其它問題,提高穩(wěn)定性。 IT之家了解,小米真無線藍牙耳機 Air 2
    的頭像 發(fā)表于 11-18 11:47 ?9072次閱讀

    小米Air 2s已修復(fù)電量10%時概率無法充電問題

    據(jù)IT之家網(wǎng)友反饋,小米 Air 2s 真無線藍牙耳機現(xiàn)已推送 2.1.5.2 版本固件,優(yōu)化了當(dāng)耳機電量為 10% 時,概率性無法充電的問題。 此前,小米 Air 2s 耳機還優(yōu)化
    的頭像 發(fā)表于 12-01 09:15 ?5148次閱讀

    Tinker Board 2S嵌入式系統(tǒng)開箱介紹

    本次取得的是Tinker Board 2S / 2GB的版本,單板微電腦就外觀來看,Tinker Board 2S與Tinker Board 2這兩個版本的各種外接界面規(guī)劃其實沒有差異
    的頭像 發(fā)表于 11-07 16:07 ?1553次閱讀

    Tinker Board 2S系統(tǒng)與軟件安裝設(shè)定

    Tinker Board 2S在官方網(wǎng)站上提供了Debian10 Linux操作系統(tǒng)給用戶進行安裝,相關(guān)LXDE桌面環(huán)境的套件軟件程序,都有相當(dāng)完整的支持,本文將會著眼在Tinker Board 2S應(yīng)用在AI范疇中
    的頭像 發(fā)表于 11-14 16:06 ?2087次閱讀