前言
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
一、反射的概述
回想一下我们以前是怎么创建对象并且使用对象中的方法或属性的
1.构造方法
设计类- >创建对象 ->调用对象的方法或属性
2.反序列化
如果仅仅知道一个类的类名,能否动态的获得类的信息,包括方法、属性?
答:通过反射可以做到
反射就是把java类中的各种成分映射成一个个的Java对象
例如:一个类有构造方法、成员方法、属性等信息,利用反射技术可以把类进行解刨,映射成一个一个对象。
Java反射相关API
1.Class 类型
2.Constructor 构造方法
3.Method 方法
4.Field 属性
除了Class外,其他类都位于java.lang.reflect包中
二、Class类
Class 类的实例表示正在运行的 Java 应用程序中的类和接口。也就是jvm中有N多的实例每个类都有该Class对象
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象.
三种方式获得类的Class对象
1.类名.class
2.对象名.getClass()
3.Class.forName(“类的地址”)
public class Car {
public String name;
public int speed;
public void start(){
System.out.println("汽车正在以"+speed+"的速度形式");
}
public void show(String location){
System.out.println("汽车正在"+location+"展览");
}
public Car(String name,Integer speed){
this.name = name;
this.speed = speed;
System.out.println("有参的构造方法");
System.out.println("我叫"+name+"我的速度是"+speed);
}
public Car() {
System.out.println("无参的构造方法");
}
}
/*反射:只知道一个类的地址,动态的获取类的信息
获取Class对象的三种方式*/
1.类名.class
Class a=Car.class;
2.getclass()方法
Class b=new Car().getClass();
3.forName()方法
String path="com.ffyc.dormitoryMS.reflect.Car";
Class c=Class.forName(path);
注意:在运行期间,一个类,只有一个Class对象产生
三、Constructor类
1.如何获得Constructor类实例
Constructor getConstructor(Class… parameterTypes) :通过指定参数类型,返回构造方法实例。
2.创建对象
con.newInstance(“zhangsan", 20);
String path="com.ffyc.dormitoryMS.reflect.Car";
Class c=Class.forName(path);
//获得指定的构造方法
Constructor con=c.getConstructor(String.class,Integer.class);
System.out.println(con.getName());
con.newInstance("宝马",100);
//返回包含一个数组 Constructor对象反射由此表示的类的所有公共构造类对象
Constructor []cons=c.getConstructors();
//所有的构造方法(包括:私有、受保护、默认、公有)
Constructor []cons2 = c.getDeclaredConstructors();
for(Constructor temp:cons){
System.out.println(temp.getParameterCount()); //获得构造方法的参数个数
System.out.println(temp.getParameterTypes()); //获得构造方法的参数类型
}
四、Field类
Field类将类的属性进行封装,可以获得属性的基本信息、属性的值,也可以对属性进行赋值。
1.getName:返回属性的名字
2.Set:设置属性值
获得Field实例,都是通过Class中的方法实现
● public Field getField(String name)
● 通过指定Field名字,返回Field实例
● 注意Field的访问权限
Class carClass = Class.forName("com.ffyc.dormitoryMS.reflect.Car");
System.out.println("************获取所有公有的字段********************");
Field[] fieldArray = carClass.getFields();
for(Field f : fieldArray){
System.out.println(f);
}
System.out.println("*******获取所有的字段(包括私有、受保护、默认的)*******");
fieldArray = carClass.getDeclaredFields();
for(Field f : fieldArray){
System.out.println(f);
}
System.out.println("*************获取公有字段并调用*********");
Field f = carClass.getField("name");
Object obj = carClass.getConstructor().newInstance(); //产生Car对象
f.set(obj, "宝马"); //设置字段值
System.out.println((Student)obj.name); //输出属性的值
四、Method类
Method实例都是通过Class类的方法获得
Method getMethod(String name, Class… parameterTypes) :通过指定方法名,参数类型,返回一个 Method实例
public class Car {
private int size;
public String name;
public int speed;
public void start(){
System.out.println("汽车正在以"+speed+"的速度形式");
}
public Car(String name,Integer speed){
this.name = name;
this.speed = speed;
System.out.println("有参的构造方法");
System.out.println("我叫"+name+"我的速度是"+speed);
}
public Car() {
System.out.println("无参的构造方法");
}
}
//1.获取Class对象
Class carClass = Class.forName("com.ffyc.dormitoryMS.reflect.Car");
//2.获取所有公有方法
System.out.println("***************获取所有的”公有“方法*******************");
carClass.getMethods();
Method[] methodArray = carClass.getMethods();
for(Method m : methodArray){
System.out.println(m);
}
System.out.println("***************获取所有的方法,包括私有的*******************");
methodArray = stuClass.getDeclaredMethods();
for(Method m : methodArray){
System.out.println(m);
}
System.out.println("***************获取公有的show()方法*******************");
Method m = carClass.getMethod("show", String.class);
//使用方法
Object obj = carClass.getConstructor().newInstance();
m.invoke(obj, "温州");
注意:使用setAccessible(true)方法可以暴力反射,解除私有限定,以Filed对象为例
Object obj = carClass.getConstructor().newInstance();
f = carClass.getDeclaredField("size");
f.setAccessible(true);//暴力反射,解除私有限定
f.set(obj, "999");
总结
反射是框架设计的灵魂,使用的前提条件:必须先得到代表的字节码的Class,Class类用于表示.class文件(字节码文件),反射的代码不是重点,重点是理解反射的过程以及在Tomcat、jdbc、框架中是如何使用反射机制来完成相对应的操作。