JAVA反射机制是在运行状态中,对于任意一个类 (class文件),都能够知道这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性,这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
简单说就是动态获取类中信息,就是java反射,可以理解为对类的解剖。
应用场景:Tomcat,我们一般来说一个应用程序为了功能的扩展都会对外提供接口,但是并不提供源码,无法new一个实例,这样的话可以通过配置文件可以获取class文件,通过反射机制来动态获取此class的内部信息。
要想要对字节码文件进行解剖,必须要有字节码文件对象,如何获取字节码文件对象呢:
方式一:
Object类中的getClass()方法的,想要用这种方式,必须要明确具体的类,并创建对象,比较麻烦。
Person p = new Person();
Class clazz = p.getClass();
Person p1 = new Person();
Class clazz1 = p1.getClass();
System.out.println(clazz==clazz1);
方式二:
任何数据类型都具备一个静态的属性.class来获取其对应的Class对象,相对简单,但是还是要明确用到类中的静态成员,还是不够扩展。
Class clazz = Person.class;
Class clazz1 = Person.class;
System.out.println(clazz==clazz1);
方式三:
只要通过给定的类的字符串名称就可以获取该类,更为扩展,可用Class类中的方法完成,该方法就是forName,这种方式只要有名称即可,更为方便,扩展性更强。
String className = "cn.itcast.bean.Person";
Class clazz = Class.forName(className);
System.out.println(clazz);
第二:获取Class中的构造函数
早期:new时候,先根据被new的类的名称找寻该类的字节码文件,并加载进内存,并创建该字节码文件对象,并接着创建该字节文件的对应的Person对象.
cn.itcast.bean.Person p = new cn.itcast.bean.Person();
现在
String name = "cn.itcast.bean.Person";
找寻该名称类文件,并加载进内存,并产生Class对象。
Class clazz = Class.forName(name);
如何产生该类的对象(实例)呢?
Object obj = clazz.newInstance();
当获取指定名称对应类中的所体现的对象时,而该对象初始化不使用空参数构造该怎么办呢?既然是通过指定的构造函数进行对象的初始化,所以应该先获取到该构造函数。通过字节码文件对象即可完成,该方法是:getConstructor(paramterTypes);
String name = "cn.itcast.bean.Person";
//找寻该名称类文件,并加载进内存,并产生Class对象。
Class clazz = Class.forName(name);
//获取到了指定的构造函数对 象。
Constructor constructor = clazz.getConstructor(String.class,int.class);
//通过该构造器对象的newInstance方法进行对象的初始化。
Object obj = constructor.newInstance("小明",38);
第三:获取Class中的字段
Class clazz = Class.forName("cn.itcast.bean.Person");
Field field = null;//clazz.getField("age");//只能获取公有的,
field = clazz.getDeclaredField("age");//只获取本类,但包含私有。
//对私有字段的访问取消权限检查。暴力访问。
field.setAccessible(true);
Object obj = clazz.newInstance();
field.set(obj, 89);
Object o = field.get(obj);
System.out.println(o);
第四:获取Class中的方法
Class clazz = Class.forName("cn.itcast.bean.Person");
Method[] methods = clazz.getMethods();//获取的都是公有的方法。
methods = clazz.getDeclaredMethods();//只获取本类中所有方法,包含私有。
for(Method method : methods){
System.out.println(method);
}
//获取空参数一般方法。
Method method = clazz.getMethod("show", null);//获取空参数一般方法。
//Object obj = clazz.newInstance();
Constructor constructor = clazz.getConstructor(String.class,int.class);
Object obj = constructor.newInstance("小明",37);
method.invoke(obj, null);
//获取带参数的一般方法
Method method = clazz.getMethod("paramMethod", String.class,int.class);
Object obj = clazz.newInstance();
method.invoke(obj, "小强",89);