由于反射涉及动态解析的类型,某些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");
method.setAccessible(Boolean.TRUE);
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去获取反射方法即可,如果已经执行过了,就直接返回,如果缓存中不存在,就查找一次后并缓存起来。

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐