最进在学习spring,在结合自己这几天的学习发现,反射机制太重要了,虽然之前自己学过,但是现在有些忘记,于是就想把Java反射机制总结以下。
反射机制
反射机制的优点
使代码更加灵活,比如在web方面,我们数据库是MySQL,我们也写了相关的dao层,但是突然我们需要换一个Orcale,但是我们完全可以通过配置文件的方式,然后再用反射机制再将加入到容器中。
反射机制的缺点
缺点也很明显,就是我们使用反射的地方将会变成解释执行,告诉JVM遇到什么情况执行什么,比直接使用要慢。
反射机制的组成
- 构造函数
- 成员变量
- 成员方法
相关代码实现
获取class对象
package com.reflect;
//测试类
public class ReflectTargetOrigin {
String defaultMember = "default";
public String publicMember = "public";
protected String protectedMember = "protected";
private String privateMember = "private";
}
public static void main(String[] args) throws ClassNotFoundException {
//方式一:通过new对象,然后调用getClass()方式
ReflectTargetOrigin reflectTargetOrigin = new ReflectTargetOrigin();
Class clazz1 = reflectTargetOrigin.getClass();
System.out.println("方式一: "+clazz1);
//方式二:通过类名.class方式获取
Class clazz2 = ReflectTargetOrigin.class;
System.out.println("方式二:"+clazz2);
//题外:方式一与方式二得到的class对象有什么关系
System.out.println("是否相等: "+(clazz1 == clazz2));
//方式三:通过Class.forName()
Class clazz3 = Class.forName("com.reflect.ReflectTargetOrigin");
System.out.println("方式三:"+clazz3);
//题外:方式二与方式三得到的class对象有什么关系
System.out.println("是否相等: "+(clazz2 == clazz3));
}
运行结果
为什么clazz1,clazz2,clazz3都是相等的呢?
这是因为,JVM中的双亲委派原则
如何实现双亲委派原则,继承ClassLoader,然后再重写findClass()。
如何打破双亲委派原则,继承ClassLoader,然后重写loadClass(),先自己尝试加载,如果不成功,在往上寻早。
获取构造函数
package com.reflect;
public class ReflectTarget extends ReflectTargetOrigin{
//---------构造函数-----------
//(默认的带参数构造函数)
ReflectTarget(String str) {
System.out.println("(默认)的构造方法 s = " + str);
}
//无参构造函数
public ReflectTarget() {
System.out.println("调用了公有的无参构造方法 。。。");
}
//有一个参数的构造函数
public ReflectTarget(char name) {
System.out.println("调用了带有一个参数的构造方法,参数值为 " + name);
}
//有多个参数的构造函数
public ReflectTarget(String name, int index) {
System.out.println("调用了带有多个参数的构造方法,参数值为【目标名】: " + name + " 【序号】" + index);
}
//受保护的构造函数
protected ReflectTarget(boolean n){
System.out.println("受保护的构造方法 n :" + n);
}
//私有的构造函数
private ReflectTarget(int index){
System.out.println("私有的构造方法 序号:" + index);
}
//**************字段*******************//
public String name;
protected int index;
char type;
private String targetInfo;
@Override
public String toString(){
return "ReflectTarget [name=" + name + ", index=" + index + ", type=" + type
+ ", targetInfo=" + targetInfo + "]";
}
//***************成员方法***************//
public void show1(String s){
System.out.println("调用了公有的,String参数的show1(): s = " + s);
}
protected void show2(){
System.out.println("调用了受保护的,无参的show2()");
}
void show3(){
System.out.println("调用了默认的,无参的show3()");
}
private String show4(int index){
System.out.println("调用了私有的,并且有返回值的,int参数的show4(): index = " + index);
return "show4result";
}
}
package com.reflect;
import java.lang.reflect.Constructor;
/*
* 通过Class对象可以获取某个类中的:构造方法;
*
* 获取构造方法:
* 1).批量的方法:
* public Constructor[] getConstructors():所有"公有的"构造方法
public Constructor[] getDeclaredConstructors():获取所有的构造方法(包括私有、受保护、默认、公有)
* 2).获取单个的方法,并调用:
* public Constructor getConstructor(Class... parameterTypes):获取单个的"公有的"构造方法:
* public Constructor getDeclaredConstructor(Class... parameterTypes):获取"某个构造方法"可以是私有的,或受保护、默认、公有;
*
* 调用构造方法:
* Constructor-->newInstance(Object... initargs)
*/
public class ConstructorCollector {
public static void main(String[] args) throws Exception {
//获取class对象
Class clazz = Class.forName("com.reflect.ReflectTarget");
//1.获取所有的公有构造方法
System.out.println("**********************所有公有构造方法*********************************");
Constructor[] conArray = clazz.getConstructors();
for(Constructor c : conArray){
System.out.println(c);
}
//2.获取所有构造方法
System.out.println("************所有的构造方法(包括:私有、受保护、默认、公有)***************");
conArray = clazz.getDeclaredConstructors();
for(Constructor c : conArray){
System.out.println(c);
}
//3.获取单个带参数的公有方法
System.out.println("*****************获取公有、有两个参数的构造方法*******************************");
Constructor con = clazz.getConstructor(String.class, int.class);
System.out.println("con = " + con);
//4.获取单个私有的构造方法
System.out.println("******************获取私有构造方法*******************************");
con = clazz.getDeclaredConstructor(int.class);
System.out.println("private con = " + con);
System.out.println("******************调用私有构造方法创建实例*******************************");
//暴力访问(忽略掉访问修饰符)
con.setAccessible(true);
ReflectTarget reflectTarget = (ReflectTarget) con.newInstance(1);
}
}
运行结果
获取成员变量
//测试用到了ReflectTargetOrigin 与 ReflectTarget
package com.reflect;
import java.lang.reflect.Field;
/*
* 获取成员变量并调用:
*
* 1.批量的
* 1).Field[] getFields():获取所有的"公有字段"
* 2).Field[] getDeclaredFields():获取所有字段,包括:私有、受保护、默认、公有;
* 2.获取单个的:
* 1).public Field getField(String fieldName):获取某个"公有的"字段;
* 2).public Field getDeclaredField(String fieldName):获取某个字段(可以是私有的)
*
* 设置字段的值:
* Field --> public void set(Object obj,Object value):
* 参数说明:
* 1.obj:要设置的字段所在的对象;
* 2.value:要为字段设置的值;
*
*/
public class FieldCollector {
public static void main(String[] args) throws Exception {
//获取Class对象
Class reflectTargetClass = Class.forName("com.reflect.ReflectTarget");
//1.获取所有公有的字段
System.out.println("************获取所有公有的字段********************");
Field[] fieldArray = reflectTargetClass.getFields();
for (Field f : fieldArray){
System.out.println(f);
}
//2.获取所有的字段
System.out.println("************获取所有的字段(包括私有、受保护、默认的)********************");
fieldArray = reflectTargetClass.getDeclaredFields();
for (Field f : fieldArray){
System.out.println(f);
}
//3.获取单个特定公有的field
System.out.println("*************获取公有字段并调用***********************************");
Field f = reflectTargetClass.getField("name");
System.out.println("公有的field name : " + f);
ReflectTarget reflectTarget = (ReflectTarget)reflectTargetClass.getConstructor().newInstance();
//4.给获取到的field赋值
f.set(reflectTarget, "待反射一号");
//5.验证对应的值name
System.out.println("验证name : " + reflectTarget.name);
//6.获取单个私有的Field
System.out.println("**************获取私有字段targetInfo并调用********************************");
f = reflectTargetClass.getDeclaredField("targetInfo");
System.out.println(f);
f.setAccessible(true);
f.set(reflectTarget, "13810592345");
System.out.println("验证信息" + reflectTarget);
}
}
运行结果
注意:
我们使用getFields()方法,是可以获取到父类的共有字段的。
获取成员函数
//测试用到了ReflectTargetOrigin 与 ReflectTarget
package com.reflect;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
/*
* 获取成员方法并调用:
*
* 1.批量的:
* public Method[] getMethods():获取所有"公有方法";(包含了父类的方法也包含Object类)
* public Method[] getDeclaredMethods():获取所有的成员方法,包括私有的(不包括继承的)
* 2.获取单个的:
* public Method getMethod(String name,Class<?>... parameterTypes):
* 参数:
* name : 方法名;
* Class ... : 形参的Class类型对象
* public Method getDeclaredMethod(String name,Class<?>... parameterTypes)
*
* 调用方法:
* Method --> public Object invoke(Object obj,Object... args):
* 参数说明:
* obj : 要调用方法的对象;
* args:调用方式时所传递的实参;
):
*/
public class MethodCollector {
public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
//1、获取Class对象
Class reflectTargetClass = Class.forName("com.reflect.ReflectTarget");
//2、获取所有公有方法
System.out.println("***************获取所有的public方法,包括父类和Object*******************");
Method[] methodArray = reflectTargetClass.getMethods();
for(Method m : methodArray){
System.out.println(m);
}
//3、获取该类的所有方法
System.out.println("***************获取所有的方法,包括私有的*******************");
methodArray = reflectTargetClass.getDeclaredMethods();
for(Method m : methodArray){
System.out.println(m);
}
//4、获取单个公有方法
System.out.println("***************获取公有的show1()方法*******************");
Method m = reflectTargetClass.getMethod("show1", String.class);
System.out.println(m);
//5、调用show1并执行
ReflectTarget reflectTarget = (ReflectTarget)reflectTargetClass.getConstructor().newInstance();
m.invoke(reflectTarget, "待反射方法一号");
//6、获取一个私有的成员方法
System.out.println("***************获取私有的show4()方法******************");
m = reflectTargetClass.getDeclaredMethod("show4", int.class);
System.out.println(m);
m.setAccessible(true);
String result = String.valueOf(m.invoke(reflectTarget, 20));
System.out.println("返回值 : " + result);
}
}
运行结果
注意
m = reflectTargetClass.getDeclaredMethod(“show4”, int.class);
System.out.println(m);
m.setAccessible(true);
String result = String.valueOf(m.invoke(reflectTarget, 20));
System.out.println("返回值 : " + result);
}
}
### 运行结果
[外链图片转存中...(img-GzHhnw4V-1590904245075)]
**注意**
getMethods():获取所有"公有方法",包含了父类的方法也包含Object类。