由于反射涉及动态解析的类型,某些Java虚拟机的优化不能被执行,所以导致了一定的性能的问题,特别是在JDK6以前特别严重,有时甚至达到数百倍,但是在JDK6以后,据说性能差别就不是哪么大了,JDK对此做了优化。但是总还是有那么数10倍的性能差异,那我们就让他跑得再快点吧。
我们通常情况下写反射都是这样写:
- SomeObject someObject = new SomeObject();
- Class<? extends SomeObject> cls = SomeObject.class;
- Method method = cls.getDeclaredMethod("someGetMethod");
- String xxx = (String) method.invoke(someObject);
这种写法从功能上来说是不会有问题的,但是从性能上来说,却是使用反射性能最差的一种方式了,每次都需要做访问检查,这个是比较消耗性能的,如果把访问检查,有人测试过,性能一下子从原来的300倍差异,降低到了只有数10倍差异:
- SomeObject someObject = new SomeObject();
- Class<? extends SomeObject> cls = SomeObject.class;
- Method method = cls.getDeclaredMethod("someGetMethod");
- <span style="color:#FF0000;"><strong>method.setAccessible(Boolean.TRUE);</strong></span>
- String xxx = (String) method.invoke(someObject);
但是即使这样,还是会涉及到每次方法的重新查找,这个也是比较耗性能的,我们可以通过把方法全部缓存起来的方式,这样只要方法被使用过一次,后面就不需要再次查找了,性能又会有一定的提升:
- public Method getMethod(String name, @SuppressWarnings("rawtypes") Class... parameterTypes) throws SecurityException, NoSuchMethodException {
- Method method = classMethodMap.get(name);//classMethodMap used to store method
- if (classMethodMap.get(name) == null) {
- method = someClass.getDeclaredMethod(name, parameterTypes);//someClass is the reflect object class
- method.setAccessible(Boolean.TRUE);
- concentrationClassMethodMap.put(name, method);
- }
- return method;
- }
然后调用的时候就只需要通过方法getMethod去获取反射方法即可,如果已经执行过了,就直接返回,如果缓存中不存在,就查找一次后并缓存起来。