Java重要概念之反射

  反射(Reflection),是指在程序运行期间,可以知道任何一个类的所有信息,可以调用任何一个对象的可供调用的方法、可供访问的字段。

1 Class类

  JVM是动态加载类的,并不是一次性加载完所有的类,而是在第一次需要用到某个类(或者接口)时,才将其编译得到的.class文件读取到内存,并且根据这个.class文件创建一个Class实例,该实例记录了这个类的所有信息。
  Class类大致如下:

public final class Class {
    private Class() {}
    ...
}

  可以看到Class类的构造函数是private修饰的,我们无法在程序中去创建Class实例。Class实例是由JVM创建的。前面说到只有第一次加载某个类时,才会创建Class实例,之后不会再创建,这也就说明一个类的Class实例在内存中是唯一的。
  通过class获取其Class实例的方法有三种:
  1.Class cls = String.class;,直接通过该class的静态变量class获得;
  2.Class cls = str.getClass();,假设我们只有一个实例,调用该实例的getClass()方法获得对应的Class实例。
  3.Class cls = Class.forName("java.lang.String");,加入我们知道一个类的完整类名,可以调用Class.forName()获得其对应的Class实例。
  获取到Class实例之后,就可以获得该类的所有信息。如果我们有一个实例,获取到它的class的信息之后,自然也可以调用其可供调用的方法、访问其可供访问的字段。
  “反射就是通过Class实例获取class的所有信息来实现的。”

2 访问字段

  获取到某个class的Class实例之后,我们可以进而获取其字段。Class类有这样几种方法:
  1.Field getField(name),根据字段名获取public的字段(包括父类);
  2.Field getDeclaredField(name),根据字段名获取当前类的字段(不包括父类);
  3.Field getFields(),获取所有public的字段(包括父类);
  4.Field getDeclaredFields(),获取当前类的所有字段(不包括父类);
  当我们获得一个Filed实例,可以通过get和set方法对字段值进行访问和修改。如果字段是private的,需要在调用set和get方法之前,调用setAccessible方法,即f.setAccessible(true)。但是如果JVM中运行着SecurityManager,可能会不允许对java和javax开头的包中的类调用setAccessible方法,以保证JVM核心库的安全。
  事实上,通过反射机制访问字段是一种非常规的办法,能够访问到private的字段,这和用private修饰字段的目的相悖。但是反射通常用在工具和底层框架,而不用在业务上。

3 调用方法

  与访问字段类似,通过Class实例的方法可以获取Method实例:getMethod(),getMethods(),getDeclaredMethod(),getDeclaredMethods();
  通过Method实例可以获取方法信息:getName(),getReturnType(),getParameterTypes(),getModifiers();
  通过Method实例可以调用某个对象的方法:Object invoke(Object instance, Object... parameters);对Method实例调用invoke方法,即相当于调用对象实例的该方法,第一个参数传入对象实例,后边传入对象方法的参数。对于静态方法,第一次参数传入null即可。
  通过设置setAccessible(true)来访问非public方法,当然也有可能被SecurityManager阻止;
  通过反射调用方法时,仍然遵循多态原则。调用的方法是实际传入invoke的对象实例类型的方法。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容