由於反射涉及動態解析的類型,某些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 methodif (classMethodMap.get(name) == null) {method = someClass.getDeclaredMethod(name, parameterTypes);//someClass is the reflect object classmethod.setAccessible(Boolean.TRUE);concentrationClassMethodMap.put(name, method);}return method;}
然後調用的時候就只需要通過方法getMethod去擷取反射方法即可,如果已經執行過了,就直接返回,如果緩衝中不存在,就尋找一次後並緩衝起來。