遇到的问题:
本地环境使用反射获取成员变量时测试正常,但将分支部署在泳道上发现利用反射拿一个类的字段时, 拿到的field数组中多了一个奇怪的变量: $jacocoData
原因:
maven集成了jacoco来统计单元测试的代码覆盖率。才会多一个 $jacocoData
(发现jacoco是用了java Agent在修改运行时字节码实现的)
对JaCoco了解不甚多, 只知道它是代码覆盖率工具. 了解了一下它的基本原理, 它使用的ASM技术修改字节码方法, 可以修改Jar文件、class文件字节码文件(详细可参考博客:JAVA代码覆盖率工具JaCoCo-原理篇-腾讯云开发者社区-腾讯云
解决方案:
使用field的 isSynthetic()方法做一次判断,跳过合成成员。
JaCoco官网FAQ中有一条是关于$jacocoData的, 如下:
https://www.eclemma.org/jacoco/trunk/doc/faq.html
官方建议是修改代码, 忽略synthetic成员.
什么是synthetic, 可参考: https://www.baeldung.com/java-synthetic
"Any constructs introduced by a Java compiler that do not have a corresponding construct in the source code must be marked as synthetic, except for default constructors, the class initialization method, and the values and valueOf methods of the Enum class."
解决方案自然是忽略synthetic成员了, 可以用下面这个方法判断是否是synthetic field:
for (Field field : fieldArray) {
// 忽略synthetic成员
if (field.isSynthetic()) {
continue;
}
...
}
所以在反射取字段时候遇到这个坑比较难排查,记录一下。这里的解决办法参考了博客:$jacocoData问题的解决,即使用了是否为复合字段的field方法解决。
总结与反思:
When your code uses reflection, please ignore synthetic members. This is a good practice anyways as also the Java compiler creates synthetic members in certain situation.