黑马程序员技术交流社区

标题: 提高Java反射速度的方法以及对setAccessable的误解 [打印本页]

作者: 张强+    时间: 2011-10-21 22:11
标题: 提高Java反射速度的方法以及对setAccessable的误解
mercyblitz 写道
ouchxp 写道
再就是在执行反射之前执行field.setAccessible(true); 也可以提高JDK反射效率
这样可以提高效率?这个方法仅仅设置访问标识,让不能访问的成员可以访问。

keating 写道
ouchxp 写道
再就是在执行反射之前执行field.setAccessible(true); 也可以提高JDK反射效率
正如楼上mercyblitz所说,如private变量...

在此澄清一下对于setAccessable的误解
先看一个例子
Java代码  
import java.lang.reflect.InvocationTargetException;   
import java.lang.reflect.Method;   
  
public class Main {   
    public static void main(String[] args) throws SecurityException, NoSuchMethodException, IllegalArgumentException, IllegalAccessException, InvocationTargetException {   
        Method m = A.class.getDeclaredMethod("getName", new Class[]{});   
        System.out.println(m.isAccessible());   
                //getName是public的,猜猜输出是true还是false   
           
        A a = new A();   
        a.setName("Mr Lee");   
        long start = System.currentTimeMillis();   
        for(int i=0;i<10000000;i++){   
            m.invoke(a, new Object[]{});   
        }   
        System.out.println( "Simple              :" +(System.currentTimeMillis() - start));   
           
        m.setAccessible(true);//注意此处不同   
        long start1 = System.currentTimeMillis();   
        for(int i=0;i<10000000;i++){   
            m.invoke(a, new Object[]{});   
        }   
        System.out.println("setAccessible(true) :"+( System.currentTimeMillis() - start1));   
    }   
}   
class A{   
    private String name;   
    public String getName() {   
        return name;   
    }   
    public void setName(String name) {   
        this.name = name;   
    }   
}
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Main {
public static void main(String[] args) throws SecurityException, NoSuchMethodException, IllegalArgumentException, IllegalAccessException, InvocationTargetException {
  Method m = A.class.getDeclaredMethod("getName", new Class[]{});
  System.out.println(m.isAccessible());
                //getName是public的,猜猜输出是true还是false
  
  A a = new A();
  a.setName("Mr Lee");
  long start = System.currentTimeMillis();
  for(int i=0;i<10000000;i++){
   m.invoke(a, new Object[]{});
  }
  System.out.println( "Simple              :" +(System.currentTimeMillis() - start));
  
  m.setAccessible(true);//注意此处不同
  long start1 = System.currentTimeMillis();
  for(int i=0;i<10000000;i++){
   m.invoke(a, new Object[]{});
  }
  System.out.println("setAccessible(true) :"+( System.currentTimeMillis() - start1));
}
}
class A{
private String name;
public String getName() {
  return name;
}
public void setName(String name) {
  this.name = name;
}
}


测试结果
引用
false
Simple              :4969
setAccessible(true) :250

明显 Accessible并不是标识方法能否访问的. public的方法 Accessible仍为false
使用了method.setAccessible(true)后 性能有了20倍的提升
Accessable属性是继承自AccessibleObject 类. 功能是启用或禁用安全检查
JDK API中的解释
引用
AccessibleObject 类是 Field、Method 和 Constructor 对象的基类。它提供了将反射的对象标记为在使用时取消默认 Java 语言访问控制检查的能力。对于公共成员、默认(打包)访问成员、受保护成员和私有成员,在分别使用 Field、Method 或 Constructor 对象来设置或获得字段、调用方法,或者创建和初始化类的新实例的时候,会执行访问检查。
在反射对象中设置 accessible 标志允许具有足够特权的复杂应用程序(比如 Java Object Serialization 或其他持久性机制)以某种通常禁止使用的方式来操作对象。
setAccessible
public void setAccessible(boolean flag)
                   throws SecurityException
将此对象的 accessible 标志设置为指示的布尔值。值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。值为 false 则指示反射的对象应该实施 Java 语言访问检查。

实际上setAccessible是启用和禁用访问安全检查的开关,并不是为true就能访问为false就不能访问
由于JDK的安全检查耗时较多.所以通过setAccessible(true)的方式关闭安全检查就可以达到提升反射速度的目的
Over




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2