大多數(shù)開發(fā)者認為性能優(yōu)化是一個復雜的話題,它需要大量的工作經(jīng)驗和相關知識理論。好吧,這也不完全錯。
優(yōu)化一個應用做到性能最優(yōu)化可能不是件容易的任務,但是這并不意味著你沒有相關的知識就什么也做不了。這里有一些易于遵循的建議和最佳實踐可以幫助你創(chuàng)建一個性能良好的應用程序。
這些建議的大部分都是針對 Java 語言的。但是也有一些是跟語言無關的,你可以運用到任意的應用和程序中。在我們學習特定的 Java 編程性能調(diào)優(yōu)之前,先來探討一些通用的技巧。
1. 在明確必要之前別急著優(yōu)化
這可能是最重要的性能優(yōu)化技巧之一。你應該遵循常見的最佳實踐做法并在案例中高效地應用它。但是這并不意味在證明必要之前,你應該更換任何標準庫或構建復雜的優(yōu)化。
多數(shù)情況下,過早地優(yōu)化會占用大量的時間,而且會使代碼變得難以理解和閱讀。更糟糕的是,這些優(yōu)化通常并沒帶來任何好處,因為你花了大量的時間在優(yōu)化應用中的非關鍵部分。
那么,要怎么證明東西需要優(yōu)化呢?
首先,你需要定義你的代碼速度得多快。例如,為所有 API 調(diào)用指定最大響應時間,或者指定在特定時間范圍內(nèi)要導入的記錄數(shù)量。在做完這些后,你需要確定你應用中哪些部分太慢需要改進。當完成這些后,你就可以來看看第二個技巧提示。
2. 使用分析器找到真正的瓶頸
在完成第一部分的優(yōu)化建議以鑒別出你應用中需要提升的部分后,要從哪里入手呢?
你可以有兩種途徑來解決這個問題:
查看你的代碼,從看起來可疑的或者你覺得可能會導致出現(xiàn)問題的地方入手。
或者使用分析器獲取代碼每個部分的行為(執(zhí)行過程)和性能的詳細信息。
希望我不需要解釋為什么應該始終遵循第二種途徑/方法的原因。
很顯然,基于分析器的方式可以讓你更好地理解代碼的性能影響,并允許你去專注于更關鍵的部分(代碼)。即使你曾經(jīng)使用過分析器,你一定記得你曾經(jīng)多么驚訝于一下就找到了代碼的哪些部分產(chǎn)生了性能問題。我第一次的猜測不止一次地導致我走錯了方向。
3. 為整個應用程序創(chuàng)建一個性能測試套件
這是另一個通用的可以幫助你避免在將性能改進部署到產(chǎn)品中之后經(jīng)常會發(fā)生的許多意外問題的技巧。你應該總是定義一個性能測試套件來測試整個應用程序,并在性能改進之前和之后運行它。
這些額外的測試運行將幫助你識別你的改動所引起的功能和性能上的副作用,并確保不會導致弊大于利的更新。如果你處理的是被應用程序的多個不同部分使用的組件,如數(shù)據(jù)庫或緩存,那這一點尤為重要。
4. 優(yōu)先關注最大瓶頸
在創(chuàng)建了測試套件并使用分析器分析你的應用程序之后,你可以列出一系列需要解決以提高性能的問題列表。這很好,但這并沒有回答你需要從哪里開始的問題。你可以專注于速成方案,或從最重要的問題開始。
速成方案一開始可能會很有吸引力,因為你可以很快顯示第一個成果。但有時,可能有必要說服其他團隊成員或管理層認為性能分析是值得的。
一般來說,我建議從頂層開始,首先開始處理最重要的性能問題。這將為你提供最大的性能改進,而且你可能僅需要解決這些問題中的一小部分就能滿足你的性能要求。
常見的通用調(diào)優(yōu)技巧到此結束。接下來讓我們仔細看看一些特定于 Java 的技巧。
5. 使用 StringBuilder 以編程方式連接字符串
在 Java 中有很多不同的選項來連接字符串。例如,你可以使用簡單的 + 或 + = ,以及老的 StringBuffer 或 StringBuilder 。
那么,你應該選擇哪種方法呢?
答案取決于連接字符串的代碼。如果你是以編程方式將新內(nèi)容添加到字符串中,例如在for循環(huán)中,則應使用 StringBuilder 。它很易于使用,并提供比 StringBuffer 更好的性能。但請記住,與 StringBuffer 相比, StringBuilder 不是線程安全的,可能并不適用于所有情況。
你只需要實例化一個新的 StringBuilder 并調(diào)用append方法來向String中添加一個新的部分。在你添加完了所有的部分后,你可以調(diào)用toString方法來檢索已連接的字符串。下面的代碼片段展示了一個簡單的例子。
在每次迭代期間,該循環(huán)將 i 轉換為一個 String ,并將其與空格一起添加到 StringBuilder sb 中。所以,最后,這段代碼在日志文件中寫入 “This is a test0 1 2 3 4 5 6 7 8 9” 。
StringBuilder sb = new StringBuilder(“This is a test”);for (int i=0; i<10; i++) { sb.append(i); sb.append(” “); } log.info(sb.toString);
正如你在代碼片段中看到的。我們可以為字符串的第一個元素提供到構造函數(shù)中。這會創(chuàng)建一個 StringBuilder ,其中包含了你所提供的字符串以及 16 個額外字符的容量。當你向 StringBuilder 中添加更多字符時,你的 JVM 將動態(tài)的增加 StringBuilder 的大小。
如果你已經(jīng)知道字符串將包含多少個字符,則可以將該數(shù)字提供給不同的構造方法以實例化具有指定容量的 StringBuilder 。這進一步提高了效率,因為它不需要動態(tài)擴展其容量。
6. 盡可能使用基本類型
避免任何開銷并提高應用程序性能的另一種簡便快速的方法是使用基本類型而不是其包裝類。所以,最好使用 int 而不是 Integer ,是 double 而不是 Double 。這將使得你的 JVM 將值存儲在堆棧而不是堆中,以減少內(nèi)存消耗,并更有效地處理它。
7. 盡量避免大整數(shù)和小數(shù)
由于我們已經(jīng)在討論數(shù)據(jù)類型,所以我們也應該快速瀏覽大整數(shù)和小數(shù)。尤其是后者因其精確性而受歡迎。但這是有代價的。大整數(shù)和小數(shù)比一個簡單的 long 型或 double 型需要更多的內(nèi)存,并會顯著減慢所有的運算。所以,如果你需要額外的精度,或者如果你的數(shù)字超出一個較長的范圍,最好要三思。這可能是你需要更改并解決性能問題的唯一方法,尤其是在實現(xiàn)數(shù)學算法時。
8. 使用 Apache Commons StringUtils.Replace 而不是 String.replace
一般來說,String.replace 方法可以正常工作,并且效率很高,尤其是在你使用 Java 9 的情況下。但是,如果你的應用程序需要大量的替換操作,并且沒有更新到最新的 Java 版本,那么檢查更快和更有效的替代品依然是有必要的。
有一種候選方案是 Apache Commons Lang 的 StringUtils.replace 方法。正如 Lukas Eder 在他最近的一篇博客文章中所描述的,它遠遠勝過了 Java 8 的 String.replace 方法。
而且它只需要很小的改動。你只需要將 Apache Commons Lang 項目的 Maven 依賴項添加到你的應用程序的 pom.xml 中,并將 String.replacemethod 的所有調(diào)用替換為 StringUtils.replace 方法。
// replace this test.replace(“test”, “simple test”); // with this StringUtils.replace(test, “test”, “simple test”);
9. 昂貴的緩存資源,如數(shù)據(jù)庫連接
緩存是避免重復執(zhí)行昂貴或常用代碼片段的流行解決方案。總的思路很簡單:重復使用這些資源比創(chuàng)建一個新的資源更劃算。
一個典型的例子是緩存池中的數(shù)據(jù)庫連接。新連接的創(chuàng)建需要時間,如果你重用現(xiàn)有連接,則可以避免這種情況。
你也可以在 Java 語言源碼中找到其他的例子。例如,在 Integer 類中的 valueOf 方法緩存了介于 -128 到 127 之間的值。你可能會說創(chuàng)建一個新的 Integer 并不是太昂貴,但是由于它經(jīng)常被使用,因此緩存最常用的值也可以提供性能優(yōu)勢。
但是,當你考慮使用緩存時,請記住緩存實現(xiàn)也會產(chǎn)生開銷。你需要花費額外的內(nèi)存來儲存可重復使用的資源,因此你可能需要管理你的緩存以使資源可訪問,并刪除過期的資源。
所以,在開始緩存任何資源之前,請確保它們是經(jīng)常使用的,以超過緩存實現(xiàn)的開銷(代價)。
總結
正如你所看到的,有時不需要太多的工作就可以提高你的應用程序的性能。本文中的大部分建議只需要稍作努力就可以將它們應用于你的代碼中。
但還是那句話,最重要的還是那些與是什么編程語言無關的技巧:
在你知道其必要性之前不要進行優(yōu)化
使用分析器(profiler)來查找真正的瓶
優(yōu)先處理最大的瓶頸
-
JAVA
+關注
關注
19文章
2978瀏覽量
105365 -
API
+關注
關注
2文章
1525瀏覽量
62546 -
代碼
+關注
關注
30文章
4846瀏覽量
69270
發(fā)布評論請先 登錄
相關推薦
xgboost超參數(shù)調(diào)優(yōu)技巧 xgboost在圖像分類中的應用
MCF8316A調(diào)優(yōu)指南

MCT8316A調(diào)優(yōu)指南

MCT8315A調(diào)優(yōu)指南

TDA3xx ISS調(diào)優(yōu)和調(diào)試基礎設施

大數(shù)據(jù)從業(yè)者必知必會的Hive SQL調(diào)優(yōu)技巧
智能調(diào)優(yōu),使步進電機安靜而高效地運行

MMC SW調(diào)優(yōu)算法

如何調(diào)優(yōu)DS160PR410實現(xiàn)出色的信號完整性

TAS58xx系列通用調(diào)優(yōu)指南

AM6xA ISP調(diào)優(yōu)指南

OSPI控制器PHY調(diào)優(yōu)算法

深度解析JVM調(diào)優(yōu)實踐應用

評論