相信很多人都知道反射可以说是Java中最强大的技术了,它可以做的事情太多太多,很多优秀的开源框架都是通过反射完成的,比如最初的很多注解框架,后来因为java反射影响性能,所以被运行时注解APT替代了,java反射有个开源框架jOOR相信很多人都用过,不过我们还是要学习反射的基础语法,这样才能自己写出优秀的框架,当然这里所讲的反射技术,是学习Android插件化技术、Hook技术等必不可少的!
Java 反射是可以让我们在运行时获取类的方法、属性、父类、接口等类的内部信息的机制。也就是说,反射本质上是一个“反着来”的过程。我们通过new创建一个类的实例时,实际上是由Java虚拟机根据这个类的Class对象在运行时构建出来的,而反射是通过一个类的Class对象来获取它的定义信息,从而我们可以访问到它的属性、方法,知道这个类的父类、实现了哪些接口等信息。
Java的 反射机制 是在运行状态中,对于任意一个类,都能够 知道这个类的所有属性和方法 ;对于任意一个对象,都能够 调用它的任意一个方法和属性 ;这种 动态获取的信息以及动态调用对象的方法的功能 称为 Java 语言的反射机制
。
简而言之,只要你给我一个 .class
——类的名字,我就能通过反射获取到类的属性和方法。
反射是很多高级技术的基础,Java 中的注解、动态代理,各种框架注入 Spring 、 MyBatis 等都用到了反射技术。
1. 在运行时判断一个对象所属的类
2. 在运行时可以构造任意一个类的对象
3. 在运行时可以任意判断一个类所包含的成员变量和方法
4. 在运行时可以任意调用一个对象的方法
5. 生成动态代理
我们知道使用javac能够将.java文件编译为.class文件,这个.class文件包含了我们对类的原始定义信息(父类、接口、构造器、属性、方法等)。.class文件在运行时会被ClassLoader加载到Java虚拟机(JVM)中,当一个.class文件被加载后,JVM会为之生成一个Class对象,我们在程序中通过new实例化的对象实际上是在运行时根据相应的Class对象构造出来的。确切的说,这个Class对象实际上是java.lang.Class泛型类的一个实例,比如Class对象即为一个封装了MyClass类的定义信息的Class实例。由于java.lang.Class类不存在公有构造器,因此我们不能直接实例化这个类,我们可以通过以下方法获取一个Class对象。
public final class Class
extends Object
implements Serializable, GenericDeclaration, Type, AnnotatedElement
复制代码
类 Class 的实例表示运行中的 Java 应用程序中的类和接口。
枚举
是一种类,注释
是一种接口。
每个 数组
也属于一个类,这个类反映为一个 类对象
,由具有相同元素类型和维数的所有数组共享。
原始Java类型(布尔型、字节型、char型、short型、int型、long型、float型和double型)和关键字void也被表示为类对象。
Class类的实例表示正在运行的Java应用程序中的类和接口。每个类只会产生一个Class对象,在类加载的时候自动创建
1 通过 class.forname()来获取Class对象
Class clazz = Class.forName("com.mashibing.entity.Emp");
System.out.println(clazz.getPackage());
System.out.println(clazz.getName());
System.out.println(clazz.getSimpleName());
System.out.println(clazz.getCanonicalName());
2 通过类名.class来获取Class对象
Class clazz = Emp.class;
System.out.println(clazz.getPackage());
System.out.println(clazz.getName());
System.out.println(clazz.getSimpleName());
System.out.println(clazz.getCanonicalName());
3 通过对象的getClass()来获取Class对象
Class clazz = new Emp().getClass();
System.out.println(clazz.getPackage());
System.out.println(clazz.getName());
System.out.println(clazz.getSimpleName());
System.out.println(clazz.getCanonicalName());
4如果是一个基本数据类型,那么可以通过Type的方式来获取Class对象
Class type = Integer.TYPE;
System.out.println(type.getName());
System.out.println(type.getCanonicalName());
获取类的构造方法:
public Constructor getConstructor(Class>... parameterTypes)
throws NoSuchMethodException,
SecurityException
public Constructor>[] getConstructors()
throws SecurityException
复制代码
获取类的成员变量
public Field getField(String name)
throws NoSuchFieldException,
SecurityException
public Field[] getFields()
throws SecurityException
复制代码
获取类的方法
public Method getMethod(String name,
Class>... parameterTypes)
throws NoSuchMethodException,
SecurityException
public Method getMethod(String name,
Class>... parameterTypes)
throws NoSuchMethodException,
SecurityException
反射在众多框架中都有普遍的应用。比如 Spring IOC
容器帮我们实例化众多的bean,下面我们简单模拟一下 反射
在其中起到的作用。
此处使用的案例接这篇:【设计模式】代理模式那些事儿:静态代理,动态代理,JDK的动态代理,cglib,Spring AOP
Spring 配置文件:
复制代码
使用的时候直接这样就能拿到定义的类了:
ApplicationContext ctx = new ClassPathXmlApplicationContext("app_aop.xml");
Pony pony = (Pony) ctx.getBean("pony");
复制代码
那么是怎么做到的呢?就是通过 反射
。
Spring 通过配置文件实例化对象,并将其放到容器的过程大概就是(模拟):
//伪代码
//1.解析 元素的id属性得到该字符串值为“pony”
String idStr = "pony";
//解析 元素的class属性得到该字符串值为“com.xblzer.dp.proxy.springaop.Pony”
String classStr = "com.xblzer.dp.proxy.springaop.Pony";
//利用反射机制,通过classStr获取Class类对象
Class> cls = Class.forName(classStr);
//实例化对象
Object obj = cls.newInstance();
//放到Spring容器
Map container = new HashMap<>();
container.put(idStr, obj);