<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          從 2s 優(yōu)化到 0.1s,這班加的值了!

          共 5010字,需瀏覽 11分鐘

           ·

          2023-11-10 12:07

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

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

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

          這其中到底經歷了什么呢?

          背 景

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

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

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

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

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

          通過這種方式,簡化了數據流程,快速把整個頁面功能調通了。

          第1次優(yōu)化

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

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

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

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

          流程圖如下:【建議收藏】細說Redis分布式鎖于是暫時這樣優(yōu)化了一下:

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

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

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

          redis常用命令和數據類型

          第2次優(yōu)化

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

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

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

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

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

          于是,流程圖改成了這樣:增加了一個job每隔5分鐘執(zhí)行一次,從數據庫中查詢分類數據,封裝成分類樹,更新到Redis緩存中。

          其他的流程保持不變。【面試官】如何解決Redis中出現的大key?

          此外,Redis的過期時間之前設置的5分鐘,現在要改成永久。

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

          第3次優(yōu)化

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

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

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

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

          該怎么優(yōu)化呢?Redis 核心篇:唯快不破的秘密

          答:加內存緩存。

          如果加了內存緩存,就需要考慮數據一致性問題。

          內存緩存是保存在服務器節(jié)點上的,不同的服務器節(jié)點更新的頻率可能有點差異,這樣可能會導致數據的不一致性。

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

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

          于是,我們使用了Spring推薦的caffine作為內存緩存。

          改造后的流程圖如下:

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

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

          這樣優(yōu)化之后,再次做網站首頁的壓力測試,qps提升到了500多,滿足上線要求。Redis數據管理進階,高效遍歷海量數據

          第4次優(yōu)化

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

          使用了很長一段時間沒有出現問題。

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

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

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

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

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

          限制分類樹的數量?

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

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

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

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

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

          第5次優(yōu)化

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

          但有一天公司同事在排查Redis中大key的時候,揪出了分類樹。之前的分類樹使用key/value的結構保存數據的。

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

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

          只保存需要用到的字段。

          例如:

          @AllArgsConstructor
          @Data
          public class Category {

              private Long id;
              private String name;
              private Long parentId;
              private Date inDate;
              private Long inUserId;
              private String inUserName;
              private List<Category> children;
          }

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

          修改自動名稱。

          例如:

          @AllArgsConstructor
          @Data
          public class Category {
              /**
               * 分類編號
               */

              @JsonProperty("i")
              private Long id;

              /**
               * 分類層級
               */

              @JsonProperty("l")
              private Integer level;

              /**
               * 分類名稱
               */

              @JsonProperty("n")
              private String name;

              /**
               * 父分類編號
               */

              @JsonProperty("p")
              private Long parentId;

              /**
               * 子分類列表
               */

              @JsonProperty("c")
              private List<Category> children;
          }

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

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

          這還不夠,需要對存儲的數據做壓縮。

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

          其實RedisTemplate支持,value保存byte數組

          先將json字符串數據用GZip工具類壓縮成byte數組,然后保存到Redis中。

          再獲取數據時,將byte數組轉換成json字符串,然后再轉換成分類樹。

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

          小 結


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


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

          ··········  END  ··············

          在看點贊轉發(fā),是對我最大的鼓勵

          瀏覽 237
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  天堂资源网在线观看 | 黄片基地 | 亚洲成人篇在线观看无码 | 日韩激情视频青青草 | 亚洲无吗中文 |