反射-框架设计的灵魂


前言

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、框架中是如何使用反射机制来完成相对应的操作。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

JinziH Never Give Up

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值