title: Java 反射机制初探
date: 2017-08-04 08:29:02
tags: [Java]
Java 反射机制
反射(Reflect)是Java语言中的重要的特性. Java框架中的绝大多数都使用了反射. 理解框架必须要精通反射的实现和原理. 在java.lang.reflect
中, Java提供了一些列反射的类. 其中, 最常用的几个类分别为Class
(这个类在java.lang
包下), Method
, Field
, Constructor
. 从字面意思上来看, 上面几个类分别对应了Java语言中的类, 方法, 字段, 构造方法. 通过学习使用这几个类, 可以让我们了解Java反射的过程和使用.
Class
Class
在java.lang
, 在Java文档中,对Class
类的描述如下
Instances of the class Class represent classes and interfaces in a running Java application. An enum is a kind of class and an annotation is a kind of interface. Every array also belongs to a class that is reflected as a Class object that is shared by all arrays with the same element type and number of dimensions. The primitive Java types (boolean, byte, char, short, int, long, float, and double), and the keyword void are also represented as Class objects.
Class has no public constructor. Instead Class objects are constructed automatically by the Java Virtual Machine as classes are loaded and by calls to the defineClass method in the class loader.
从上面我们可以看到Class
是Java类的对象. 从Class
的实例中, 我们可以操纵Java指定的类. 而且Class
类没有构造函数, 只能从一些特定途径生成类对象.
下面代码展示了如何获取一个类对象,首先我们声明一个Student
类, 该类是一个普通的类. 下面的所有代码不外加说明的话, 都在此类的基础上展开.
package reflect;
public class Student {
private int sId;
private int sName;
public int getsId() {
return sId;
}
public void setsId(int sId) {
this.sId = sId;
}
public int getsName() {
return sName;
}
public void setsName(int sName) {
this.sName = sName;
}
private void say(){
System.out.println("hello");
}
public void play(){
System.out.println("play");
}
public Student() {
}
public Student(int id){
this.sId = id;
}
public Student(int id, int sname){
this.sId = id;
this.sName = sname;
}
}
- 使用类全称获取类对象
Class clazz = Class.forName("reflect.Student");
- 使用类名获取类对象
Class clazz = Student.class
- 使用实例获得类对象
Student s = new Student();
Class clazz = s.getClass();
上述三种方式都可以获取Student
的Class
对象. 目前来说, 第一种使用类名的字符串获取类对象更加常用.
在一个类对象中有一个方法特别值得注意:newInstance()
. 这个方法从字面上就可以很好理解: 产生一个该类的实例.
Class clazz = Class.forName("reflect.Student");
Object o = clazz.newInstance();
但是这个方法只能调用无参构造函数. 当需要调用带参构造函数时或者类中没有无参构造函数时, 就不能用了. 幸好Java提供了Constructor
类去调用其他的构造函数.
Constructor
Constructor
在java.lang.reflect
包中. 在Java文档中,对Constructor
类的描述如下
Constructor provides information about, and access to, a single constructor for a class.
Constructor permits widening conversions to occur when matching the actual parameters to newInstance() with the underlying constructor's formal parameters, but throws an IllegalArgumentException if a narrowing conversion would occur.
正如上面所说, Constructor
是为了解决调用带参构造函数的时候使用, 当然它也可以用来调用无参构造函数. Constructor
相当于Class
中的newInstance()
函数的升级版本.
具体用法如下
Constructor[] constructors = clazz.getConstructors();
for(int i=0;i<constructors.length;i++){
Method getIdMethod = clazz.getMethod("getsId");
System.out.println(constructors[i].getName()+" "+constructors[i].getParameterCount());
if(constructors[i].getParameterCount()==0){
Object o = constructors[i].newInstance();
System.out.println(getIdMethod.invoke(o));
}
else if(constructors[i].getParameterCount()==1){
Object o = constructors[i].newInstance(1);
System.out.println(getIdMethod.invoke(o));
}else{
Object o = constructors[i].newInstance(2,3);
System.out.println(getIdMethod.invoke(o));
}
}
Method
Method
在java.lang.reflect
包中. 在Java文档中,对Method
类的描述如下:
A Method provides information about, and access to, a single method on a class or interface. The reflected method may be a class method or an instance method (including an abstract method).
A Method permits widening conversions to occur when matching the actual parameters to invoke with the underlying method's formal parameters, but it throws an IllegalArgumentException if a narrowing conversion would occur.
Method
类是对Java类方法的抽象. 具体使用方法和示例代码如下
Method playMethod = clazz.getMethod("play");
playMethod.invoke(clazz.newInstance());
Method[] allMethod = clazz.getDeclaredMethods();
for(int i=0;i<allMethod.length;i++){
System.out.println(allMethod[i].getName()+" "+allMethod[i].getParameterCount());
}
Method sayMethod = clazz.getDeclaredMethod("say");
sayMethod.setAccessible(true);
sayMethod.invoke(clazz.newInstance());
sayMethod.setAccessible(false);
注意到这句话sayMethod.setAccessible(true)
, 目的是让调用者有权限去执行类中的非public方法(或者权限不够的方法).
Field
Field
在java.lang.reflect
包中. 在Java文档中,对Field
类的描述如下:
A Field provides information about, and dynamic access to, a single field of a class or an interface. The reflected field may be a class (static) field or an instance field.
A Field permits widening conversions to occur during a get or set access operation, but throws an IllegalArgumentException if a narrowing conversion would occur.
Field
类是对Java类字段的抽象. 具体使用方法和示例代码如下
Field[] fields = clazz.getDeclaredFields();
for(int i=0;i<fields.length;i++){
System.out.println(fields[i].getName());
}
Object o = clazz.newInstance();
Method m = clazz.getMethod("setsId", int.class);
System.out.println(m.getName());
m.invoke(o, 1);
Field sidField = clazz.getDeclaredField("sId");
System.out.println(sidField.getInt(o));
以上就是Java反射中的四个比较重要和基础的类. reflect
包中还有许多重要的类, 比方说AOP的实现工具InvocationHandler
动态代理等等. 理解基本的反射工具可以让我们更好地了解Java语言的特性和Java框架的原理.