使用JMH(Java Microbench Mark Marness)對(duì)Java代碼進(jìn)行基準(zhǔn)測(cè)試標(biāo)準(zhǔn)
Jul 27, 2025 am 02:40 AM不能手寫基準(zhǔn)測(cè)試是因?yàn)镴VM的優(yōu)化機(jī)制(如死代碼消除、JIT編譯)會(huì)導(dǎo)致結(jié)果失真;1. 使用JMH需添加jmh-core和jmh-generator-annprocess依賴;2. 用@Benchmark註解標(biāo)記測(cè)試方法並使用Blackhole防止結(jié)果被優(yōu)化;3. 通過@BenchmarkMode、@Warmup、@Measurement、@Fork、@State等註解合理配置測(cè)試環(huán)境;4. 運(yùn)行時(shí)通過main方法啟動(dòng)JMH,避免手動(dòng)循環(huán)測(cè)試,確保結(jié)果準(zhǔn)確可信。
使用JMH(Java Microbenchmark Harness)對(duì)Java 代碼進(jìn)行基準(zhǔn)測(cè)試,是準(zhǔn)確衡量小段代碼性能的標(biāo)準(zhǔn)方式。很多人直接用System.currentTimeMillis()
或簡(jiǎn)單循環(huán)測(cè)試性能,但這種方式容易受到JIT 編譯、GC、CPU 預(yù)熱、代碼重排序等JVM 優(yōu)化機(jī)制的影響,導(dǎo)致結(jié)果嚴(yán)重失真。 JMH 就是為了解決這些問題而生的。

為什麼不能手寫基準(zhǔn)測(cè)試?
JVM 不是簡(jiǎn)單的解釋器,它有JIT 編譯、方法內(nèi)聯(lián)、逃逸分析、死代碼消除等優(yōu)化。例如:
long start = System.nanoTime(); for (int i = 0; i < 1000; i ) { someMethod(); } long end = System.nanoTime();
JVM 可能會(huì)發(fā)現(xiàn)someMethod()
的結(jié)果沒被使用,直接優(yōu)化掉整個(gè)循環(huán)(死代碼消除),導(dǎo)致測(cè)出來耗時(shí)為0。這種“假快”毫無意義。

如何正確使用JMH?
1. 添加JMH 依賴(Maven 示例)
<dependency> <groupId>org.openjdk.jmh</groupId> <artifactId>jmh-core</artifactId> <version>1.37</version> </dependency> <dependency> <groupId>org.openjdk.jmh</groupId> <artifactId>jmh-generator-annprocess</artifactId> <version>1.37</version> <scope>provided</scope> </dependency>
2. 創(chuàng)建基準(zhǔn)測(cè)試類
@Benchmark public void testStringConcat(Blackhole blackhole) { String s = ""; for (int i = 0; i < 10; i ) { s = "a"; } blackhole.consume(s); // 防止被優(yōu)化掉} @Benchmark public void testStringBuilder() { StringBuilder sb = new StringBuilder(); for (int i = 0; i < 10; i ) { sb.append("a"); } sb.toString(); }
關(guān)鍵點(diǎn):
- 使用
@Benchmark
註解標(biāo)記測(cè)試方法。 - 使用
Blackhole
防止結(jié)果被JIT 優(yōu)化掉(尤其是返回值或中間變量未被使用時(shí))。 - 方法必須是
public
,否則JMH 無法訪問。
3. 添加合理的註解控制測(cè)試行為
@BenchmarkMode(Mode.AverageTime) @OutputTimeUnit(TimeUnit.NANOSECONDS) @Warmup(iterations = 3, time = 1, timeUnit = TimeUnit.SECONDS) @Measurement(iterations = 5, time = 1, timeUnit = TimeUnit.SECONDS) @Fork(1) @State(Scope.Thread) public class StringConcatBenchmark { // 測(cè)試方法寫在這裡}
說明:

-
@BenchmarkMode
: 測(cè)平均耗時(shí)(AverageTime
)、吞吐量(Throughput
)等。 -
@Warmup
: 預(yù)熱輪次,讓JIT 充分優(yōu)化代碼。 -
@Measurement
: 實(shí)際測(cè)量輪次。 -
@Fork(1)
: 每次測(cè)試fork 一個(gè)新JVM 進(jìn)程,避免相互干擾。 -
@State(Scope.Thread)
: 表示測(cè)試狀態(tài)的作用域,避免線程間干擾。
4. 運(yùn)行基準(zhǔn)測(cè)試
寫一個(gè)main
方法啟動(dòng):
public class JmhRunner { public static void main(String[] args) throws Exception { org.openjdk.jmh.Main.main(args); } }
然後運(yùn)行:
mvn clean install java -cp target/benchmarks.jar com.example.StringConcatBenchmark
或者直接在IDE 中運(yùn)行main
方法(確保已生成JMH 類)。
常見陷阱與最佳實(shí)踐
- ?不要手動(dòng)寫
for
循環(huán)測(cè)性能:JMH 已經(jīng)幫你控制迭代次數(shù)和測(cè)量邏輯。 - ?使用
Blackhole
消費(fèi)結(jié)果:防止無用代碼被優(yōu)化。 - ?避免在
@Setup
中做太重的操作:除非是測(cè)試初始化開銷。 - ?關(guān)注預(yù)熱(Warmup)是否充分:JIT 編譯需要時(shí)間,預(yù)熱不足會(huì)導(dǎo)致結(jié)果偏慢。
- ?多次運(yùn)行取穩(wěn)定值:性能可能受系統(tǒng)負(fù)載影響,建議多次測(cè)試。
示例輸出解讀
Benchmark Mode Cnt Score Error Units StringConcatBenchmark.testStringConcat avgt 5 482.3 ± 12.7 ns/op StringConcatBenchmark.testStringBuilder avgt 5 32.1 ± 1.3 ns/op
-
avgt
: 平均耗時(shí) -
ns/op
: 每次操作納秒數(shù) -
Score
: 平均值,Error
: 誤差範(fàn)圍
可以看出StringBuilder
比字符串拼接快一個(gè)數(shù)量級(jí)。
基本上就這些。 JMH 的核心價(jià)值是幫你避開JVM 的“陷阱”,拿到真實(shí)反映性能的數(shù)據(jù)。只要按規(guī)範(fàn)寫,結(jié)果就可信。不復(fù)雜,但容易忽略細(xì)節(jié)。
以上是使用JMH(Java Microbench Mark Marness)對(duì)Java代碼進(jìn)行基準(zhǔn)測(cè)試標(biāo)準(zhǔn)的詳細(xì)內(nèi)容。更多資訊請(qǐng)關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

熱AI工具

Undress AI Tool
免費(fèi)脫衣圖片

Undresser.AI Undress
人工智慧驅(qū)動(dòng)的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費(fèi)的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費(fèi)的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強(qiáng)大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6
視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版
神級(jí)程式碼編輯軟體(SublimeText3)

Callable和Runnable在Java中主要有三點(diǎn)區(qū)別。第一,Callable的call()方法可以返回結(jié)果,適合需要返回值的任務(wù),如Callable;而Runnable的run()方法無返回值,適用於無需返回的任務(wù),如日誌記錄。第二,Callable允許拋出checked異常,便於錯(cuò)誤傳遞;而Runnable必須在內(nèi)部處理異常。第三,Runnable可直接傳給Thread或ExecutorService,而Callable只能提交給ExecutorService,並返回Future對(duì)像以

Java支持異步編程的方式包括使用CompletableFuture、響應(yīng)式流(如ProjectReactor)以及Java19 中的虛擬線程。 1.CompletableFuture通過鍊式調(diào)用提升代碼可讀性和維護(hù)性,支持任務(wù)編排和異常處理;2.ProjectReactor提供Mono和Flux類型實(shí)現(xiàn)響應(yīng)式編程,具備背壓機(jī)制和豐富的操作符;3.虛擬線程減少並發(fā)成本,適用於I/O密集型任務(wù),與傳統(tǒng)平臺(tái)線程相比更輕量且易於擴(kuò)展。每種方式均有適用場(chǎng)景,應(yīng)根據(jù)需求選擇合適工具並避免混合模型以保持簡(jiǎn)潔性

JavaNIO是Java1.4引入的新型IOAPI,1)面向緩衝區(qū)和通道,2)包含Buffer、Channel和Selector核心組件,3)支持非阻塞模式,4)相比傳統(tǒng)IO更高效處理並發(fā)連接。其優(yōu)勢(shì)體現(xiàn)在:1)非阻塞IO減少線程開銷,2)Buffer提升數(shù)據(jù)傳輸效率,3)Selector實(shí)現(xiàn)多路復(fù)用,4)內(nèi)存映射加快文件讀寫。使用時(shí)需注意:1)Buffer的flip/clear操作易混淆,2)非阻塞下需手動(dòng)處理不完整數(shù)據(jù),3)Selector註冊(cè)需及時(shí)取消,4)NIO並非適用於所有場(chǎng)景。

在Java中,枚舉(enum)適合表示固定常量集合,最佳實(shí)踐包括:1.用enum表示固定狀態(tài)或選項(xiàng),提升類型安全和可讀性;2.為枚舉添加屬性和方法以增強(qiáng)靈活性,如定義字段、構(gòu)造函數(shù)、輔助方法等;3.使用EnumMap和EnumSet提高性能和類型安全性,因其基於數(shù)組實(shí)現(xiàn)更高效;4.避免濫用enum,如動(dòng)態(tài)值、頻繁變更或複雜邏輯場(chǎng)景應(yīng)使用其他方式替代。正確使用enum能提升代碼質(zhì)量並減少錯(cuò)誤,但需注意其適用邊界。

Java的類加載機(jī)制通過ClassLoader實(shí)現(xiàn),其核心工作流程分為加載、鏈接和初始化三個(gè)階段。加載階段由ClassLoader動(dòng)態(tài)讀取類的字節(jié)碼並創(chuàng)建Class對(duì)象;鏈接包括驗(yàn)證類的正確性、為靜態(tài)變量分配內(nèi)存及解析符號(hào)引用;初始化則執(zhí)行靜態(tài)代碼塊和靜態(tài)變量賦值。類加載採(cǎi)用雙親委派模型,優(yōu)先委託父類加載器查找類,依次嘗試Bootstrap、Extension和ApplicationClassLoader,確保核心類庫(kù)安全且避免重複加載。開發(fā)者可自定義ClassLoader,如URLClassL

Javaprovidesmultiplesynchronizationtoolsforthreadsafety.1.synchronizedblocksensuremutualexclusionbylockingmethodsorspecificcodesections.2.ReentrantLockoffersadvancedcontrol,includingtryLockandfairnesspolicies.3.Conditionvariablesallowthreadstowaitfor

Java異常處理的關(guān)鍵在於區(qū)分checked和unchecked異常並合理使用try-catch、finally及日誌記錄。 1.checked異常如IOException需強(qiáng)制處理,適用於可預(yù)期的外部問題;2.unchecked異常如NullPointerException通常由程序邏輯錯(cuò)誤引起,屬於運(yùn)行時(shí)錯(cuò)誤;3.捕獲異常時(shí)應(yīng)具體明確,避免籠統(tǒng)捕獲Exception;4.推薦使用try-with-resources自動(dòng)關(guān)閉資源,減少手動(dòng)清理代碼;5.異常處理中應(yīng)結(jié)合日誌框架記錄詳細(xì)信息,便於後

HashMap在Java中通過哈希表實(shí)現(xiàn)鍵值對(duì)存儲(chǔ),其核心在於快速定位數(shù)據(jù)位置。 1.首先使用鍵的hashCode()方法生成哈希值,並通過位運(yùn)算轉(zhuǎn)換為數(shù)組索引;2.不同對(duì)象可能產(chǎn)生相同哈希值,導(dǎo)致衝突,此時(shí)以鍊錶形式掛載節(jié)點(diǎn),JDK8後鍊錶過長(zhǎng)(默認(rèn)長(zhǎng)度8)則轉(zhuǎn)為紅黑樹提升效率;3.使用自定義類作鍵時(shí)必須重寫equals()和hashCode()方法;4.HashMap動(dòng)態(tài)擴(kuò)容,當(dāng)元素?cái)?shù)超過容量乘以負(fù)載因子(默認(rèn)0.75)時(shí),擴(kuò)容並重新哈希;5.HashMap非線程安全,多線程下應(yīng)使用Concu
