Java反射

编译时类型和运行时类型

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)。

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值