编译时类型和运行时类型
Java程序中的对象在运行时都会出现两种类型:编译时类型和运行时类型。
如多态中,
Person p=new Student();
这行代码将会生成一个p变量,该变量的编译类型为Person,运行时类型为Strudent;
为了解决这些问题,程序需要在运行时发现对象和类的真实信息。为了解决这个问题,有两个做法:
- 第一种是假设在编译和运行时都完全知道类型的具体信息,这种情况下直接使用instanceof运算符进行判断,再利用强制类型装换将其装换成运行时类型的变量。
- 第二种时在编译时无法预知该对象和类可能属于哪些类,程序只能依靠运行时信息来发现对象和类的真实信息。则使用反射。
获得Class对象
每个类被加载后,系统会为该类生成一个对应的Class对象,通过该Class对象就可以访问到JVM中的的这个类。Java获得Class对象的方式有三种。
1. 使用Class类的静态方法forName.
static Class<?> forName(String className)
static Class<?> forName(String name, boolean initialize, ClassLoader loader)
说明:className是类的全限定类名。当类找不到,该方法抛出ClassNotFoundException异常。
2. 调用某个类的class属性来获得类对象的Class对象。
3. 调用对象的getClass()方法,该方法时java.lang.Object类的方法。该方法返回该对象所属类对应的Class对象。
第二种方式比第一种方式的优势:
1. 代码更安全,程序在编译阶段就可以检查需要访问的Class对象是否存在。
2. 程序性能更高,这种方式无须调用方法性能更好。
从Class中获取信息
以下四个方法访问Class对应类的构造器:
Constructor<T> getConstructor(Class<?>... parameterTypes) :返回此Class对象所表示的类的指定public构造器。
Constructor<?>[] getConstructors() :返回此Class对象所表示类的所有public构造器。
Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) :返回此Class对象所表示的类的指定构造器。无视访问级别。
Constructor<?>[] getDeclaredConstructors() :返回此Class对象所表示类的所有构造器,无视访问级别。
说明:getDeclaredConstructor,getDeclaredConstructors方法无视访问级别。即可以访问对象的private,protected,public的构造方法。
以下四个方法访问Class对应类或接口的方法:
Method getMethod(String name, Class<?>... parameterTypes)
Method[] getMethods()
Method getDeclaredMethod(String name, Class<?>... parameterTypes)
Method[] getDeclaredMethods()
说明:以上方法返回Class对应类或接口声明的指定方法方法,不包括继承的方法。
例如:
某类
public class MyObj{
public void fun(){}
public void fun(String x){}
public void fun(String x,int p){}
}
要获取该类的fun(String x,int p)方法。
则首先得到MyObj类的Class对象。
Class clazz=MyObj.class;
clazz.getMethod(“fun”,String.class,Integer.class)
以下四个方法访问Class对应类或接口的字段:
Field getField(String name)
Field[] getFields()
Method getDeclaredMethod(String name, Class<?>... parameterTypes)
Method[] getDeclaredMethods()
说明:
返回 Field 对象。反映此 Class 对象所表示的类或接口所声明的字段。
以下三个方法访问Class对应类或接口的注释:
<A extends Annotation> A getAnnotation(Class<A> annotationClass) :试图获取该Class对象所表示类上指定类型的注释:如果该类型的注释不存在则返回null。
Annotation[] getAnnotations() :返回元素上存在的所有注释。
Annotation[] getDeclaredAnnotations() :返回直接存在于此元素上的所有注释。
如下方法用于访问该Class对象对应类包含的内部类:
如下方法用于访问该Class对象对应类所实现的内部类:
如下几个方法判类是否为接口,枚举,注释类型等:
使用反射操作对象
创建对象
使用反射生成对象有两种方法:
方法一:访问默认构造方法,来创建该类的实例。
使用Class对象的newInstance()方法来创建Class对象对应类的实例。
T newInstance()
方法二:选择使用类的构造器,来创建该类的实例。
先使用Class对象获取指定的Constructor对象,再调用Constructor的newInstance()方法来创建该Class对象对应类的实例。
T newInstance(Object... initargs)
该方法参数部分传入构造方法需要的参数。
调用方法
1. 得到类对应的Class对象。
2. 得到指定的Method对象。
3. Method类包含invoke方法,用以调用该方法:
Object invoke(Object obj, Object... args) :
该方法中的obj是执行该方法的主调方法,args是执行该方法时传入该方法的实数。
若通过Method对象调用类的private方法, 则需要使Method对象获取调用该方法的权限:
public static void setAccessible(boolean flag)
将Method对象的accessible标志设置为指示的布尔值。若值为true,则无视方法的访问权限。若值为false,则不能随意访问方法。该方法可以取消访问权限检查,从而让程序访问private方法,private属性。
访问属性值
得到类对应的Class对象后,得到指定的Field对象。
Field类包含两种方法访问属性:
getXxx(Object obj) :获取obj对象该Field的属性值。此处Xxx对应8个基本类型。如果该属性的类型是引用类型,则该方法变为get(Object obj)。
setXxx(Object obj,Xxx val) :将obj对象的该Field设置成val值。此处Xxx对应8个基本类型。如果该属性的类型是引用类型,则该方法变为set(Object obj,Object val)。
操作数组
java.lang.reflect包下还提供Array类,Array对象可以代表所有的数组。程序可以通过使用Array来动态创建数组,操作数组。
static Object newInstance(Class<?> componentType, int... dimensions) : 创建一个具有指定的元素类型,指定维度的新数组。
static Object newInstance(Class<?> componentType, int length) :创建一个具有指定的元素类型,指定长度的新数组。
注意:新数组的维数不能超过该实现所支持的数组维数(通常为 255)。
static xxx getXxx(Object array,int index) :返回array数组中序号为index的元素。其中xxx是各种基本数据类型,如果数组元素是引用类型,则该方法变为get(Object array,int index)。
static void setXxx(Object obj,int index,Xxx val) :将array数组中序号为index的元素设为val。其中xxx是各种基本数据类型,如果数组元素是引用类型,则该方法变为set(Object array,int index,Onject val)。