很早就听说了大名鼎鼎的反射机制,最近终于把它学了,总结一下Java反射机制的学习。
1.获得Class类。在JVM中,加载一个类的时候,会在堆中生成该类对应的Class类,通过Class类可以获得该类的所有属性和方法,Class类就像镜子一样反射着该类的所有内容。所以,在使用Java反射机制的时候,第一步就是获得该类的Class类(Class类位于java.lang包下面),获得Class类有以下几种方法:
1)Class.forName(类的全限定名);如:Class<?> clazz = Class.forName("java.lang.String");
2) 通过getClass()方法;getClass是Object类的方法,通过类的任一实例调用getClass()。
3)运用getSuperClass(),获得父类的Class类。
4).class语法:Java的八种基本类型,类名,数组,clazz = String.class;clazz = int.class;clazz = int[].class;
5) .TYPE语法:八种基本类型的包装类型和Void,clazz = Integer.TYPE;clazz = Void.TYPE;
2.通过反射机制获得该类的实例。得到Class类之后,有以下方法可以获得该类的实例,假设D类。
1)运行Class类的newInstance()方法,相当于调用无参构造方法:D d = clazz.newInstance();
2)运用getDeclaredConstructor()方法。由于newInstance(方法只能调用无参构造方法,局限性太大。
public Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes)
该方法的参数是构造方法参数类型所对应的Class类,假设D类有构造方法:public D(String name,int age){}。那么获得该构造方法所对应的类:
Constructor cst = clazz.getDeclaredConstructor(new Class[]{String.class,int.class});
Constructor类有一个实例对象的方法newInstance(Object...params),参数列表和构造方法对应
D d = (D) cst.newInstance("cxy",25);//或者cst.newInstance(new Object[]{"cxy",25});
3.通过Java反射机制完成对类的私有成员变量的访问和修改
类中的成员变量对应的类是Field,方法对应的类是Method,通过它们可以完成对成员变量和方法的操作。下面演示如何通过Java的反射机制完成对类的私有成员变量的访问和修改(这个对我的冲击挺大的),解释都在代码中了:
package com.ssy.reflection;
import java.lang.reflect.Field;
/*
* 该类演示了如果通过java的反射机制获得和修改类的私有变量
*/
public class ChangePrivate {
/**
* 通过Class获得成员变量name所对应的Field类,Field的父类是AccessibleObject,它有一个方法,setAccessible(boolean flag)
* 如果flag=true,就取消访问控制检查,这样即使name是私有的,我们也能去访问它
* 注意
*/
public static void main(String[] args)throws Exception {
PrivateT pt = new PrivateT();
//获得对应的Class类
Class<?> clazz = pt.getClass();
//获得name字段对应的类
Field field = clazz.getDeclaredField("name");
field.setAccessible(true);//取消访问控制检查
//获得name的值,get()的参数表示获取哪个实例的值
String name = (String) field.get(pt);
System.out.println(name);
//修改name的值,set(Object obj,Object value),obj表示修改哪个实例的name,value表示修改的值
field.set(pt, "xlf");
System.out.println(pt.getName());
}
}
class PrivateT {
private String name = "cxy";
public String getName(){
return this.name;
}
}
最后在说一句:反射是的前提是运行时,不是编译时,也就说以上的所有全部是在运行时完成的。