偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

是什么讓一段20行代碼的性能提升了10倍

原創(chuàng) 精選
開發(fā) 前端
支付寶卡包存放著用戶的會員卡和優(yōu)惠券。無論是卡券cell,還是卡券詳情,都是通過靜態(tài)模板配置加上動態(tài)可變數(shù)據(jù),最終呈現(xiàn)給終端用戶的。

作者 | 金盛杰(司旭)

一、背景

1.1 業(yè)務背景

支付寶卡包存放著用戶的會員卡和優(yōu)惠券。無論是卡券cell,還是卡券詳情,都是通過靜態(tài)模板配置加上動態(tài)可變數(shù)據(jù),最終呈現(xiàn)給終端用戶的。

下面【圖1】展現(xiàn)了卡券數(shù)據(jù)在C端用戶的展現(xiàn)形式,【圖2】表示了C端數(shù)據(jù)組裝過程。

?

圖片

?【圖1】卡券數(shù)據(jù)在C端展現(xiàn)形式

圖片

??

【圖2】C端數(shù)據(jù)組裝過程

以【圖2】為例,模板中有availableAmount 和voucherName 兩個變量,這兩個變量在動態(tài)變量數(shù)據(jù)有對應的值。用動態(tài)的值替換掉模板里面對應的這兩個變量,最后拼裝成“100元紅包名稱”。當這個紅包被使用了一次,消費了30元后,動態(tài)數(shù)據(jù)里面availableAmount 的值就會變成70。用戶再次進入到紅包詳情頁時,展現(xiàn)數(shù)據(jù)重新組裝后就會變成“70元紅包名稱”。?

1.2 問題發(fā)現(xiàn)

最近做項目過程中,把卡券組裝渲染邏輯好好的梳理了一遍,其中仔細研讀了【圖3】這段模板變量替換邏輯。這是一段老代碼,從卡包產(chǎn)品誕生之日起就存在,差不多有十年的時間了。其作用就是用動態(tài)數(shù)據(jù)替換掉模板里面的變量。這段代碼邏輯咋一看,并沒有什么問題,就是把模板里面兩個$ 之間(包含)的變量,用動態(tài)數(shù)據(jù)進行替換。考慮到這是一段極為核心又高頻的調(diào)用邏輯,于是看看有沒有性能優(yōu)化的空間。

圖片

【圖3】模板變量替換代碼實現(xiàn)

把替換邏輯厘清了之后,第一感覺就是這段代碼有性能提升的空間。主要有兩點:

  • 每次while 循環(huán)進行了兩次indexOf 操作
  • 每次while 循環(huán)都進行了substring 操作

于是,就有了下面兩個疑問:

  • 能夠減少indexOf 和substring 操作嗎?
  • 真的每次都要進行模板變量查找嗎?

二、性能優(yōu)化

帶著上面兩個問題,逐步進行性能優(yōu)化并測試。

整個優(yōu)化過程一共迭代了5版,并最終取得了性能提升超過10倍的效果。下面分別來介紹下不同版本的實現(xiàn)和性能對比。?

2.1 性能優(yōu)化V1

這一版去掉了indexOf 和substring 操作,轉(zhuǎn)而使用另一種替換方式。

之前的替換邏輯是從頭到尾循環(huán)模板內(nèi)容字符串,遇到$ 之間的變量就進行替換,過程中需要不斷的進行indexOf 和substring 操作。新的實現(xiàn)方式是在進行變量替換之前,通過循環(huán)模板內(nèi)容字符串,利用雙指針把模板里面所有變量都提取出來,再對變量集合進行循環(huán),依次替換掉模板內(nèi)容里面的變量。

圖片

【圖4】性能優(yōu)化V1代碼實現(xiàn)?

2.2 性能優(yōu)化V2

靜態(tài)模板配置一般情況下不會發(fā)生變更。也就意味著,同一個模板對應的變量都是固定不變的??梢詫⒛0錳d和模板變量集合進行一對一的緩存,減少每次替換之前的變量提取。

在決定使用緩存之前,要想好怎么實現(xiàn)緩存。有兩點需要注意:

  • 用本地緩存代替TBase,減少大流量場景下對TBase的壓力
  • 么控制本地緩存的有效數(shù)量,并在有限的內(nèi)存占用情況下最大化緩存效率

可以借助Google Guava庫的緩存類來實現(xiàn)緩存邏輯,示例代碼見【圖5】

??

圖片

【圖5】緩存實現(xiàn)示例代碼

圖片

??【圖6】性能優(yōu)化V2代碼實現(xiàn)

2.3 性能對比(1)

做完上面兩步之后進行了性能測試,性能對比如【圖7】所示。

圖片

【圖7】V1、V2版性能對比

通過性能對比發(fā)現(xiàn),V1版相對于原始版有性能提升,帶緩存的V2版相對于不帶緩存的V1版也有性能提升。但隨著流量增大,性能優(yōu)化效果逐步減弱。說明V1、V2版耗時優(yōu)化的點,在整個模板變量替換耗時中占比并不高。也同時說明,整個模板變量替換邏輯當中,還存在其他更為耗時的點。

回過頭來再仔細看一遍變量替換邏輯,突然間意識到遺漏了一個”大問題“。就是這個String.replace 方法,該方法有兩個耗時點:

  • 每次replace 都會進行模板編譯
  • replace 都是創(chuàng)建一個新的對象進行返回

并且每次replace 之后還要進行變量的重新賦值。

圖片

??

【圖8】String.replace 代碼實現(xiàn)

2.4 性能優(yōu)化V3

在V2版基礎上,去掉replace  方法,用StringBuilder  來實現(xiàn)。

??

圖片

【圖9】性能優(yōu)化V3代碼實現(xiàn)

StringBuilder 實現(xiàn)過程中有一點要注意。V2版本中,提取變量返回的是一個Set 集合。返回集合中出現(xiàn)變量的順序和模板中變量順序會不一致,模板中有多個相同變量的情況下,也只會替換第一個出現(xiàn)的變量。所以要將變量提取返回的結果換成有序可重復的List ,才能保證邏輯的正確性。

2.5 性能優(yōu)化V4

V3版優(yōu)化之后,性能提升明顯,證明String.replace 方法才是整個模板變量替換邏輯中最為耗時的點。于是在原方法上只用StringBuilder 來替換String.replace ,得到V4版。

??

圖片

【圖10】性能優(yōu)化V4代碼實現(xiàn)?

2.6 性能對比(2)

圖片

【圖11】V1、V2、V3、V4版性能對比

通過【圖11】可以明顯的發(fā)現(xiàn),在進行StringBuilder 實現(xiàn)后,性能提升超過10倍,效果十分明顯。?

V4版耗時實際上比V3版帶緩存的還要少,說明V3版先提取變量再進行StringBuilder 組裝的過程,相對來說還是會更耗時一點。但V4版的代碼可讀性是不如V3版的,可以把V3版和V4版相結合,剔除掉緩存依賴,產(chǎn)生一個代碼可讀性和性能最佳的V5版。

2.7 性能優(yōu)化V5

先提取變量,去掉緩存依賴,用StringBuilder 替換掉String.replace ,增加代碼可讀性。

??

圖片

【圖12】V5版代碼實現(xiàn)&100萬次循環(huán)耗時對比

三、總結

通過上面5個版本的性能優(yōu)化,性能得到了超過10倍的提升。?

性能由高到低的順序是V4 > V3 > V5 > V2 > V1 > 未被優(yōu)化的原始版。其中V3、V4、V5版的性能顯著優(yōu)于V1和V2版,證明這段模板替換邏輯最為耗時的點為String.replace ,V3 > V5和V2 > V1表明,引入緩存對性能提升還是有一定幫助的。在代碼可讀性方面,V4是不如V3和V5的。

整個優(yōu)化總結下來主要有兩點:

1、String.replace 方法涉及到模板編譯和新字符串生成,比較吃資源

2、StringBuilder 代替String.replace ,除了能夠縮短調(diào)用耗時,在空間上也能夠減少資源占用。因為StringBuilder.append 相對于String.replace 來說,能夠減少中間大量String 對象的創(chuàng)建和銷毀,能夠減少GC的壓力,從而降低CPU的負載。

性能優(yōu)化顯而易見的好處是能夠節(jié)約機器資源。如果一個有2000臺服務器的應用,整體性能提升了10%,理論上來說,就相當于節(jié)省了200臺的機器。除了節(jié)省機器資源外,性能好的應用相對于性能差的應用,在應對流量突增時更不容易達到機器的性能瓶頸,在同樣流量場景下進行機器擴容時,也只需要更少的機器,從而能夠更快的完成擴容、應急操作。所以,性能好的應用相對于性能差的應用在穩(wěn)定性方面也更勝一籌。

最后再回到本次文章的主題:是什么讓一段20行代碼的性能提升了10倍?

我的回答是:StringBuilder yyds!?

責任編輯:武曉燕 來源: 阿里開發(fā)者
相關推薦

2020-07-22 08:30:02

代碼開發(fā)工具

2024-10-29 08:21:05

2025-05-27 01:55:00

TypeScript開發(fā)者項目

2020-03-26 12:38:15

代碼節(jié)點數(shù)據(jù)

2024-12-13 13:58:53

2025-05-09 02:00:00

代碼接口吞吐量

2024-07-17 08:25:44

2020-07-21 15:40:55

NginxJava服務器

2021-02-02 15:38:19

Disruptor緩存Java

2011-07-01 10:11:39

2023-10-20 08:12:00

JDK21線程池配置

2022-04-21 07:51:51

場景JavaSQL

2014-03-26 10:00:06

RailsRails性能

2021-09-13 10:25:35

開發(fā)技能代碼

2022-09-21 17:43:29

Kafka底層網(wǎng)絡

2023-03-22 13:53:26

芯片英偉達

2022-09-27 18:19:32

Java數(shù)據(jù)結構

2021-08-02 10:50:57

性能微服務數(shù)據(jù)

2024-09-03 09:08:43

2021-12-29 11:06:25

Java代碼技巧
點贊
收藏

51CTO技術棧公眾號