一、反射的概念
- Java反射机制的核心是在程序运行时动态加载类并获取类的详细信息,从而操作类或对象的属性和方法。本质是JVM得到class对象之后,再通过class对象进行反编译,从而获取对象的各种信息。
- Java属于先编译再运行的语言,程序中对象的类型在编译期就确定下来了,而当程序在运行时可能需要动态加载某些类,这些类因为之前用不到,所以没有被加载到JVM。通过反射,可以在运行时动态地创建对象并调用其属性,不需要提前在编译期知道运行的对象是谁。
二、反射的原理
加载过程:
正常类加载过程
1. new Student; //当程序执行到此处创建对象时,会触发JVM调用AppClassLoader加载Student.class文件
2. JVM 从磁盘中找到Student.class文件并加载到JVM内存中
3. 当.class文件加载进内存后,JVM自动创建一个Class对象。(根据双亲委派模型,可得一个类只会产生一个class对象)
4. 每一次创建对象是基于第一次产生的class对象
Java反射本质
因为在双亲委派模型下每个类只加载一次,所以,可以根据class对象反向获取创建的对象的各种信息
三、反射的优缺点:
1、优点:在运行时获得类的各种内容,进行反编译,对于Java这种先编译再运行的语言,能够让我们很方便的创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代码的链接,更加容易实现面向对象。
2、缺点:(1)反射会消耗一定的系统资源,因此,如果不需要动态地创建一个对象,那么就不需要用反射;
(2)反射调用方法时可以忽略权限检查,因此可能会破坏封装性而导致安全问题。
四、反射的用途
- 反编译:.class文件 反编译尾 .java文件
- 访问并获取对象的属性,方法等
- 通用框架的部分功能的实现(Spring中的Bean,IOC的底层DI的实现)
- 反射机制常用的类:
- Java.lang.Class;
- Java.lang.reflect.Constructor;
- Java.lang.reflect.Field;
- Java.lang.reflect.Method;
- Java.lang.reflect.Modifier;
五、反射的基本使用:
-
获得Class:主要有三种方法:
(1)Object–>getClass
obj.getClass()
(2)任何数据类型(包括基本的数据类型)都有一个“静态”的class属性
Class stuClass2 = Student.class
(3)通过class类的静态方法:forName(String className)(最常用)
Class.forName("类的全限定名")
-
创建实例:通过反射来生产对象主要的两种方法
- 使用Class对象的newInstance()方法来创建Class对象对应类的实例。
Class<?> c = String.class; //获取类对象 Object str = c.newInstance(); //通过类对象创建实例
- 先通过Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建对象,这种方法可以用指定的构造器构造类的实例。
Class<?> c = String.class; //获取类对象
Constructor constructor=c.getConstructor(String.class); //通过Class对象获取指定Constructor构造器对象
Object str = c.newInstance(); //通过类对象创建实例
可以通过isInstance()方法来判断为某个类,进行类型转换
-
反射机制的方法
- public Constructor[] getConstructors():所有"公有的"构造方法
- public Constructor[] getDeclaredConstructors():获取所有的构造方法(包括私有、受保护、默认、公有)
- public Constructor getConstructor(Class… parameterTypes):获取单个的"公有的"构造方法;
- public Constructor getDeclaredConstructor(Class… parameterTypes):获取"某个构造方法"可以是私有的,或受保护、默认、公有;
- Constructor–>newInstance(Object… initargs) //调用构造方法
- (class)obj.get属性值 //获取类属性
- objClass.getMethods();
- stuClass.getDeclaredMethod() //暴力获取私有方法 //通过setAccessible(true);//解除私有限定
- invoke(obj, 20);//需要两个参数,一个是要调用的对象(获取有反射),一个是实参
-
反射方法的其他使用–通过反射越过泛型检查
泛型用在编译期,编译过后泛型擦除(消失掉),所以是可以通过反射越过泛型检查的
-
利用反射创建数值:
Class<?> cls = Class.forName("java.lang.String"); //数组的类型 Object array = Array.newInstance(cls,25);
-
反射方法的其他使用–通过反射运行配置文件内容