4年前 (2020-11-05)  Java系列 |   抢沙发  468 
文章评分 0 次,平均分 0.0

在Java中,普遍认为java.reflect API的性能代价很高。旧的Java版本有巨大的性能开销,而新版本似乎使其处于可接受的范围内。但“可接受”到底是什么意思呢?

这是我在评论一个建议用标准代码替换基于反射的代码的性能评估时提出的问题。由于我们的许多决策不是基于事实而是基于信念,所以我决定执行一些测试来获得Java8中的指标。

测试协议

为了通过一个不受挑战的协议获得实际的度量,我使用了优秀的JMH测试框架。JMH的优势包括:

  • Maven已经存在了
  • 要进行基准测试的方法只需使用@benchmark进行注释
  • 它处理JVM的预热
  • 它还处理在控制台上写入结果的操作

以下是JMH的片段:

@Benchmark
public void executePerformanceTest() {
	// Code goes here
}

JMH将负责执行上述executePerformanceTest()并负责测量所花费的时间。

反射代码

为了强调反射的成本,让我们检查一下使用反射访问属性和不使用反射调用简单getter所需的时间差异。

// With reflection
Field firstName = clazz.getDeclaredField("firstName");
Field lastName = clazz.getDeclaredField("lastName");
Field birthDate = clazz.getDeclaredField("birthDate");
Field.setAccessible(new AccessibleObject[] { firstName, lastName, birthDate }, true);
firstName.get(person);
lastName.get(person);
birthDate.get(person);

// Without reflection
person.getFirstName();
person.getLastName();
person.getBirthDate();

检查可能的优化

我想知道不可变的数据结构是否被编译成优化的字节码,从而降低反射的性能开销。

因此,我用两种不同的方式创建了相同的基本数据结构:

  • 一个带有无参数构造函数和setter的变量
  • 一个具有最终属性和构造函数初始化的不可变

反射结果

在我的机器上运行测试会产生以下结果:

# Run complete. Total time: 00:26:55

Benchmark                                    Mode  Cnt         Score        Error  Units
BenchmarkRun.runImmutableWithReflection     thrpt  200   2492673.501 ±  37994.941  ops/s
BenchmarkRun.runImmutableWithoutReflection  thrpt  200  26499946.587 ± 242499.198  ops/s
BenchmarkRun.runMutableWithReflection       thrpt  200   2505239.277 ±  27697.028  ops/s
BenchmarkRun.runMutableWithoutReflection    thrpt  200  26635097.050 ± 150798.911  ops/s

Java反射的性能成本

对于对条形图感兴趣的读者(请注意,刻度是线性的):

Java反射的性能成本

关于反射性能的测试还可以参考这篇文章:https://javakk.com/733.html

 

除特别注明外,本站所有文章均为老K的Java博客原创,转载请注明出处来自https://javakk.com/768.html

关于

发表评论

表情 格式

暂无评论

登录

忘记密码 ?

切换登录

注册