参考:
反射例程:http://www.cnblogs.com/rollenholt/archive/2011/09/02/2163758.html
反射:https://www.jianshu.com/p/f67182a482eb
https://www.jianshu.com/p/779b3e27b26d
反射和动态代理:https://www.jianshu.com/p/27b255a6f479
https://www.cnblogs.com/hanganglin/p/4485999.html
一.概述
1.反射定义
- 在运行状态中
1)对于任意一个类,都能够知道这个类中的所有方法和属性
2)对于任意一个对象,都能够调用它的任意一个方法和属性 - 这种动态获取类的信息以及动态调用对象方法的功能称为Java语言的反射机制
2.反射的优点和缺点
-
静态编译和动态编译的概念
1)静态编译:在编译时确定类型或绑定对象
2)动态编译:在运行时确定类型或绑定对象。动态编译最大限度发挥了Java的灵活性,体现多态的应用,降低类之间的耦合性 - 优点:可以实现动态创建对象和编译,比较灵活
场景:大型软件难以一次性设计得很完美,如果使用静态编译,需要卸载原有程序,将整个程序重新编译,而采用反射机制可避免重新安装程序,在程序运行时动态编译更新的类和对象即可 - 缺点:反射是一种解释操作,对性能有影响
3.反射的应用场景
- 逆向代码,如反编译
- 与注解向结合的框架,如Retrofit
- 单纯的反射机制应用框架,如EventBus
- 动态生成类的框架,如Gson
二.反射的使用
1.Class类和类类型(class type)
- 类是
java.lang.Class
类的实例对象 -
Class
是所有类的类 - 类类型是类的类型,描述一个类是什么,有哪些方法和属性
- 反射通过类类型调用一个类的属性和方法
- 获取一个类的类类型方法有:
1)通过类的静态成员变量class:Class c1 = Demo.class;
2)通过类的对象的getClass()方法:Class c2 = demo1.getClass();
3)Class类的静态方法forName():Class c3 = Class.forName("com.dengxin.reflection.Demo");
2.通过反射查看类信息
- 获取类类型
//第一种方式 通过Class类的静态方法——forName()来实现
Class<?> class1 = Class.forName("com.dengxin.reflection.Demo");
//第二种方式 通过类的class属性
Class<?> class1 = Demo.class;
//第三种方式 通过对象getClass方法
Demo demo = new Demo();
Class<?> class1 = demo.getClass();
- 通过类类型获取class对象的成员变量
Field[] allFields = class1.getDeclaredFields(); //获取class对象的所有属性
Field[] publicFields = class1.getFields(); //获取class对象的public属性
Field ageField = class1.getDeclaredField("age"); //获取class指定属性
Field desField = class1.getField("des"); //获取class指定的public属性
- 通过类类型获取class对象的方法
//获取class对象的所有声明方法
Method[] methods = class1.getDeclaredMethods();
//获取class对象的所有public方法 包括父类的方法
Method[] allMethods = class1.getMethods();
//返回次Class对象对应类的、带指定形参列表的public方法
Method method = class1.getMethod("info", String.class);
//返回次Class对象对应类的、带指定形参列表的方法
Method declaredMethod = class1.getDeclaredMethod("info", String.class);
- 通过类类型获取class对象的构造方法
//获取class对象的所有声明构造函数
Constructor<?>[] allConstructors = class1.getDeclaredConstructors();
//获取class对象public构造函数
Constructor<?>[] publicConstructors = class1.getConstructors();
//获取指定声明构造函数
Constructor<?> constructor = class1.getDeclaredConstructor(String.class);
//获取指定声明的public构造函数
Constructor publicConstructor = class1.getConstructor(String.class);
- 通过类类型获取class对象的信息
oolean isPrimitive = class1.isPrimitive(); //判断是否是基础类型
boolean isArray = class1.isArray(); //判断是否是数组
boolean isAnnotation = class1.isAnnotation(); //判断是否是注解类
boolean isInterface = class1.isInterface(); //判断是否是接口类
boolean isEnum = class1.isEnum(); //判断是否是枚举类
boolean isAnonymousClass = class1.isAnonymousClass(); //判断是否是匿名内部类
boolean isAnnotationPresent = class1.isAnnotationPresent(Deprecated.class); //判断是否被某个注解类修饰
String className = class1.getName(); //获取class名字 包含包名路径
Package aPackage = class1.getPackage(); //获取class的包信息
String simpleName = class1.getSimpleName(); //获取class类名
int modifiers = class1.getModifiers(); //获取class访问权限
Class<?>[] declaredClasses = class1.getDeclaredClasses(); //内部类
Class<?> declaringClass = class1.getDeclaringClass(); //外部类
- 其他方法
Annotation[] annotations = (Annotation[]) class1.getAnnotations(); //获取class对象的所有注解
Annotation annotation = (Annotation) class1.getAnnotation(Deprecated.class); //获取class对象指定注解
Type genericSuperclass = class1.getGenericSuperclass(); //获取class对象的直接超类的 Type
Type[] interfaceTypes = class1.getGenericInterfaces(); //获取class对象的所有接口的type集合
3.通过反射生成并操作对象
- 生成类的实例对象,两种方法
1)使用Class对象的newInstance()
方法创建该Class对象对应类的实例。此方法要求该Class对象的对应类有默认构造器
2)使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()
方法创建Class对象对应类的实例。通过该方法可选择使用指定的构造器创建实例
//第一种方式 Class对象调用newInstance()方法生成
Object obj = class1.newInstance();
//第二种方式 对象获得对应的Constructor对象,再通过该Constructor对象的newInstance()方法生成
Constructor<?> constructor = class1.getDeclaredConstructor(String.class);
obj = constructor.newInstance("hello");
- 调用类的方法
1)先通过Class对象的getMethods()
方法或者getMethod()
方法返回Method数组或对象
2)调用Method对象的*Object invoke(Object obj, Object... args)*
方法,第一个参数为调用该方法的实例对象,第二个参数为该方法的参数
3)通过Method的invoke()
方法调用对象方法时,要求必须有方法权限。可通过调用Method的setAccessible(true)
取消调用方法的访问权限检查
// 生成新的对象:用newInstance()方法
Object obj = class1.newInstance();
//首先需要获得与该方法对应的Method对象
Method method = class1.getDeclaredMethod("setAge", int.class);
//调用指定的函数并传递参数
method.invoke(obj, 28);
三.反射与动态代理
1.代理概述
- 定义
给某个对象提供一个代理对象,并由代理对象控制对原对象的访问。即客户不直接操控原对象,而是通过代理间接地操控原对象 - 场景
1)因为安全需要屏蔽客户端直接访问真实对象
2)在远程调用中使用代理类处理远程方法调用的技术细节
3)为了提升性能,封装真实对象,达到延迟加载的目的 - 代理模式的参与者
1)主题接口:Subject
是原对象和代理对象都共同实现的接口,即代理类所实现的行为接口,Request()
是委托对象和代理对象共同拥有的方法
2)目标对象:RealSubject
是原对象,被Proxy
所代理
3)代理对象:Proxy
是代理对象,用来封装原对象
4)客户端:通过主题接口和代理对象完成业务,不直接接触目标对象
- 代理模式的实现思路
1)代理对象和目标对象均实现同一行为接口
2)代理对象和目标对象分别实现行为接口的方法逻辑
3)在代理类的构造函数中实例化一个目标对象,并通过这个对象调用目标对象的行为接口
4)客户端只能通过代理类实现的目标对象,间接调用目标对象的行为接口 - 分类
静态代理:代理类在编译时实现,即Java编译完成后代理类是一个实际的class文件
动态代理:代理类在运行时生成,即Java在运行时动态生成类字节码并加载到JVM中 - 静态代理的简单实现
public class ProxyDemo {
public static void main(String args[]){
RealSubject subject = new RealSubject();
Proxy p = new Proxy(subject);
p.request();
}
}
//主题接口
interface Subject{
void request();
}
//目标对象
class RealSubject implements Subject{
public void request(){
System.out.println("request");
}
}
//代理对象
class Proxy implements Subject{
private Subject subject;
public Proxy(Subject subject){
this.subject = subject;
}
public void request(){
System.out.println("PreProcess");
subject.request();
System.out.println("PostProcess");
}
}
2.动态代理
-
优点
1)不需要写一个与目标对象RealSubject
形式完全一样的封装类,代理对象不必实现公共接口Subject
的每个方法定义,从而减少接口变动引起的代码修改量
2)动态代理生成的方法甚至可以在运行时制定代理类的执行逻辑,从而大大提升系统灵活性 -
动态代理的分类
1)JDK动态代理:包含Proxy类和InvocationHandler接口,只能代理实现接口的类
2)Cglib动态代理:针对类实现代理,对指定的目标类生成一个子类,并通过覆盖其中方法实现增强,采用继承实现,无法代理被final修饰的类,使用Cglib需下载cglib.jar -
JDK动态代理相关类
1)java.lang.reflect.Proxy类
提供用户创建动态代理类和代理对象的静态方法,是所有动态代理类的父类
2)java.lang.reflect.InvocationHandler接口
代理对象通过实现InvocationHandler
接口的invoke()
方法调用目标对象方法 -
Proxy创建动态代理对象的两种方法
//方法1:通过动态代理类的构造方法创造
//创建一个InvocationHandler对象
InvocationHandler handler = new MyInvocationHandler(.args..);
//使用Proxy生成一个动态代理类
Class proxyClass = Proxy.getProxyClass(RealSubject.class.getClassLoader(),RealSubject.class.getInterfaces(), handler);
//获取proxyClass类中一个带InvocationHandler参数的构造器
Constructor constructor = proxyClass.getConstructor(InvocationHandler.class);
//调用constructor的newInstance方法来创建动态代理实例
RealSubject real = (RealSubject)constructor.newInstance(handler);
//方法2:通过Proxy类的newProxyInstance()静态方法直接创造
//创建一个InvocationHandler对象
InvocationHandler handler = new MyInvocationHandler(.args..);
//使用Proxy直接生成一个动态代理对象
RealSubject real =Proxy.newProxyInstance(RealSubject.class.getClassLoader(),RealSubject.class.getInterfaces(), handler);
- 其他常用方法
1)InvocationHandler getInvocationHandler(Object proxy)
Proxy类方法,获取代理对象对应的调用处理器对象
2)Class getProxyClass(ClassLoader loader,Class[] interfaces)
Proxy类方法,根据类加载器和实现的接口获得代理类
3)invoke(Object proxy,Method method,Object[] args)
InvocationHandler接口方法,调用代理对象绑定的真实对象的方法。 参数说明:① Object proxy:指被代理的对象;② Method method:要调用的方法;③ Object[] args:方法调用时所需要的参数; - JDK动态代理的简单实现
public class DynamicProxyDemo {
public static void main(String[] args) {
//1.创建目标对象
RealSubject realSubject = new RealSubject();
//2.创建调用处理器对象
ProxyHandler handler = new ProxyHandler(realSubject);
//3.动态生成代理对象
Subject proxySubject = (Subject)Proxy.newProxyInstance(
RealSubject.class.getClassLoader(),RealSubject.class.getInterfaces(), handler);
//4.通过代理对象调用方法
proxySubject.request();
}
}
/**
* 主题接口
*/
interface Subject{
void request();
}
/**
* 目标对象类
*/
class RealSubject implements Subject{
public void request(){
System.out.println("====RealSubject Request====");
}
}
/**
* 代理类的调用处理器
*/
class ProxyHandler implements InvocationHandler{
private Subject subject;
public ProxyHandler(Subject subject){
this.subject = subject;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
//定义预处理的工作,当然你也可以根据 method 的不同进行不同的预处理工作
System.out.println("====before====");
//调用RealSubject中的方法
Object result = method.invoke(subject, args);
System.out.println("====after====");
return result;
}
}
月度银墙,不辨花丛哪瓣香
网友评论