java反射的性能問題


 

java反射效率到底如何,花了點時間,做了一個簡單的測試.供大家參考.

測試背景
1. 測試簡單Bean(int,Integer,String)set方法
2. loop 1億次
3. 測試代碼盡可能避免對象的創建,復發方法的調用,僅僅測試set方法的耗時

測試結果:

 場景

 本機測試結果(XP,雙核,2G)

服務器測試結果(Linux,XEN虛擬機,8,5.5G)

方法直接調用

235MS

190MS

JDK Method調用

29188MS

4633MS

JDK Method調用(稍作優化)

5672MS

4262MS

Cglib FastMethod調用

5390MS

2787MS


得出一個感性的結果
1.JDK反射效率是直接調用的一個數量級,差不多20
2.一個set方法的反射調用時間 = 4633ms / 1/ 3= 0.0154us
3.Cglibfastmethod還是有優勢的

 

 

 

 

 

 

 

 

當然反射不止一種方法的,而且也有一些比較常見的優化方式。我們將會測試一下:

直接訪問的耗時

直接反射的耗時

緩存需要查找的函數反射的耗時

使用reflectasm的反射耗時

 

long now;long sum = 0;

TestClass t = new TestClass();

 

now = System.currentTimeMillis();

 for(int i = 0; i<500000; ++i){

    t.setNum(i);

    sum += t.getNum();

}

 

System.out.println("get-set耗時"+(System.currentTimeMillis() - now) + "ms秒,和是" +sum);

 

sum = 0;

now = System.currentTimeMillis();

 for(int i = 0; i<500000; ++i){

    Class<?> c = Class.forName("test.TestClass");

    Class<?>[] argsType = new Class[1];

    argsType[0] = int.class;

    Method m = c.getMethod("setNum", argsType);

    m.invoke(t, i);

    sum += t.getNum();

}

System.out.println("標准反射耗時"+(System.currentTimeMillis() - now) + "ms,和是" +sum);

 

sum = 0;

 

Class<?> c = Class.forName("test.TestClass");

Class<?>[] argsType = new Class[1];

argsType[0] = int.class;

Method m = c.getMethod("setNum", argsType);

 

now = System.currentTimeMillis();

 for(int i = 0; i<500000; ++i){

    m.invoke(t, i);

    sum += t.getNum();

}

System.out.println("緩存反射耗時"+(System.currentTimeMillis() - now) + "ms,和是" +sum);

 

sum = 0;

MethodAccess ma = MethodAccess.get(TestClass.class);int index = ma.getIndex("setNum");

now = System.currentTimeMillis();

 for(int i = 0; i<500000; ++i){

    ma.invoke(t, index, i);

    sum += t.getNum();

}

System.out.println("reflectasm反射耗時"+(System.currentTimeMillis() - now) + "ms,和是" +sum);

 

測試結果如下: 
get-set耗時6ms秒,和是124999750000
標准反射耗時1838ms,和是124999750000
緩存反射耗時70ms,和是124999750000
reflectasm反射耗時20ms,和是124999750000


可以看出,查找函數依然是耗時最長的部分,JDK7的優化確實很不錯,由JDK640倍降到10倍左右,reflectasm invoke的效率比java原生invoke好,大致是直接訪問的4倍時間。效率確實可以一用。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM