Java 反射到底慢在哪?( 三 )

沿用上面的测试方法 , 测试结果如下:

Java 反射到底慢在哪?

文章插图
 
invoke和set
修改getReflectCallMethodCostTime和getReflectCallFieldCostTime方法的代码如下 , 对getMethod和getDeclaredField进行测试:
private long getReflectCallMethodCostTime(int count){    long startTime = System.currentTimeMillis();    ProgramMonkey programMonkey = new ProgramMonkey("小明", "男", 12);    for(int index = 0 ; index < count; index++){        try{            Method setmLanguageMethod = programMonkey.getClass().getMethod("setmLanguage", String.class);        }catch(NoSuchMethodException e){            e.printStackTrace();        }    }    return System.currentTimeMillis()-startTime;}private long getReflectCallFieldCostTime(int count){    long startTime = System.currentTimeMillis();    ProgramMonkey programMonkey = new ProgramMonkey("小明", "男", 12);    for(int index = 0 ; index < count; index++){        try{            Field ageField = programMonkey.getClass().getDeclaredField("mLanguage");        }catch(NoSuchFieldException e){            e.printStackTrace();        }    }    return System.currentTimeMillis()-startTime;}沿用上面的测试方法 , 测试结果如下:
Java 反射到底慢在哪?

文章插图
 
getMethod和getDeclaredField
测试结论:
  • getMethod和getDeclaredField方法会比invoke和set方法耗时;
  • 随着测试数量级越大 , 性能差异的比例越趋于稳定;
由于测试的这四个方法最终调用的都是native方法 , 无法进一步跟踪 。
个人猜测应该是和在程序运行时操作class有关 , 比如需要判断是否安全?是否允许这样操作?入参是否正确?是否能够在虚拟机中找到需要反射的类?主要是这一系列判断条件导致了反射耗时;也有可能是因为调用natvie方法 , 需要使用JNI接口 , 导致了性能问题(参照Log.java、System.out.println , 都是调用native方法 , 重复调用多次耗时很明显) 。
如果避免反射导致的性能问题?通过上面的测试可以看出 , 过多地使用反射 , 的确会存在性能问题 , 但如果使用得当 , 所谓反射导致性能问题也就不是问题了 , 关于反射对性能的影响 , 参照下面的使用原则 , 并不会有什么明显的问题:
  • 不要过于频繁地使用反射 , 大量地使用反射会带来性能问题;
  • 通过反射直接访问实例会比访问方法快很多 , 所以应该优先采用访问实例的方式 。
后记上面的测试并不全面 , 但在一定程度上能够反映出反射的确会导致性能问题 , 也能够大概知道是哪个地方导致的问题 。如果后面有必要进一步测试 , 我会从下面几个方面作进一步测试:
  • 测试频繁调用native方法是否会有明显的性能问题;
  • 测试同一个方法内 , 过多的条件判断是否会有明显的性能问题;
  • 测试类的复杂程度是否会对反射的性能有明显影响 。
来自:https://www.jianshu.com/p/4e2b49fa8ba1

【Java 反射到底慢在哪?】


推荐阅读