其实通过代码我们可以发现,是Class.forName这个方法比较耗时,它实际上调用了一个本地方法,通过这个方法来要求JVM查找并加载指定的类。所以我们在项目中使用的时候,可以把Class.forName返回的Class对象缓存起来,下一次使用的时候直接从缓存里面获取,这样就极大的提高了获取Class的效率。同理,在我们获取Constructor、Method等对象的时候也可以缓存起来使用,避免每次使用时再来耗费时间创建。
测试反射调用方法
- @Test
- public void testReflexMethod() throws Exception {
- long start = System.currentTimeMillis();
- Class testUserClass = Class.forName("RefleDemo.TestUser");
- TestUser testUser = (TestUser) testUserClass.newInstance();
- Method method = testUserClass.getMethod("sayHi");
- int i = 0;
- while(i<100000000){
- ++i;
- method.invoke(testUser);
- }
- long end = System.currentTimeMillis();
- System.out.println("反射调用方法耗时:"+(end - start ) + "ms");
- }
- //反射调用方法耗时:330ms
- @Test
- public void testReflexMethod() throws Exception {
- long start = System.currentTimeMillis();
- Class testUserClass = Class.forName("RefleDemo.TestUser");
- TestUser testUser = (TestUser) testUserClass.newInstance();
- Method method = testUserClass.getMethod("sayHi");
- int i = 0;
- while(i<100000000){
- ++i;
- method.setAccessible(true);
- method.invoke(testUser);
- }
- long end = System.currentTimeMillis();
- System.out.println("setAccessible=true 反射调用方法耗时:"+(end - start ) + "ms");
- }
- //setAccessible=true 反射调用方法耗时:188ms
这里我们反射调用sayHi方法1亿次,在调用了method.setAccessible(true)后,发现快了将近一半。查看API可以了解到,jdk在设置获取字段,调用方法的时候会执行安全访问检查,而此类操作会比较耗时,所以通过setAccessible(true)的方式可以关闭安全检查,从而提升反射效率。
极致的反射
除了上面的手段,还有没有什么办法可以更极致的使用反射呢?这里介绍一个高性能反射工具包ReflectASM。它是通过字节码生成的方式来实现的反射机制,下面是一个跟java反射的性能比较。
这里就不介绍它的用法了,有兴趣的朋友可以直接传送过去:https://github.com/EsotericSoftware/reflectasm
结语
(编辑:ASP站长网)
|