Java JDK 动态代理使用及实现原理分析.docx
《Java JDK 动态代理使用及实现原理分析.docx》由会员分享,可在线阅读,更多相关《Java JDK 动态代理使用及实现原理分析.docx(27页珍藏版)》请在三一办公上搜索。
1、Java JDK 动态代理使用及实现原理分析Java JDK 动态代理使用及实现原理分析 一、什么是代理? 代理是一种常用的设计模式,其目的就是为其他对象提供一个代理以控制对某个对象的访问。代理类负责为委托类预处理消息,过滤消息并转发消息,以及进行消息被委托类执行后的后续处理。 代理模式UML图: 为了保持行为的一致性,代理类和委托类通常会实现相同的接口,所以在访问者看来两者没有丝毫的区别。通过代理类这中间一层,能有效控制对委托类对象的直接访问,也可以很好地隐藏和保护委托类对象,同时也为实施不同控制策略预留了空间,从而在设计上获得了更大的灵活性。Java 动态代理机制以巧妙的方式近乎完美地实践
2、了代理模式的设计理念。 二、Java 动态代理类 Java动态代理类位于java.lang.reflect包下,一般主要涉及到以下两个类: (1)Interface InvocationHandler:该接口中仅定义了一个方法 java view plain copy 在CODE上查看代码片派生到我的代码片 publicobject invoke(Object obj,Method method, Object args) 在实际使用时,第一个参数obj一般是指代理类,method是被代理的方法,如上例中的request,args为该方法的参数数组。这个抽象方法在代理类中动态实现。 (2)Pr
3、oxy:该类即为动态代理类,其中主要包含以下内容: protected Proxy(InvocationHandler h):构造函数,用于给内部的h赋值。 static Class getProxyClass (ClassLoaderloader, Class interfaces):获得一个代理类,其中loader是类装载器,interfaces是真实类所拥有的全部接口的数组。 static Object newProxyInstance(ClassLoaderloader, Class interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类
4、可以当作被代理类使用(可使用被代理类的在Subject接口中声明过的方法) 所谓DynamicProxy是这样一种class:它是在运行时生成的class,在生成它时你必须提供一组interface给它,然后该class就宣称它实现了这些 interface。你当然可以把该class的实例当作这些interface中的任何一个来用。当然,这个DynamicProxy其实就是一个Proxy,它不会替你作实质性的工作,在生成它的实例时你必须提供一个handler,由它接管实际的工作。 在使用动态代理类时,我们必须实现InvocationHandler接口 通过这种方式,被代理的对象(RealSub
5、ject)可以在运行时动态改变,需要控制的接口(Subject接口)可以在运行时改变,控制的方式(DynamicSubject类)也可以动态改变,从而实现了非常灵活的动态代理关系。 动态代理步骤: 1.创建一个实现接口InvocationHandler的类,它必须实现invoke方法 2.创建被代理的类以及接口 3.通过Proxy的静态方法 newProxyInstance(ClassLoaderloader, Class interfaces, InvocationHandler h)创建一个代理 4.通过代理调用方法 三、JDK的动态代理怎么使用? 1、需要动态代理的接口: java vi
6、ew plain copy 在CODE上查看代码片派生到我的代码片 package jiankunking; /* * 需要动态代理的接口 */ public interface Subject /* * 你好 * * param name * return */ public String SayHello(String name); /* * 再见 * * return */ public String SayGoodBye; 2、需要代理的实际对象 java view plain copy 在CODE上查看代码片派生到我的代码片 package jiankunking; /* * 实际对
7、象 */ public class RealSubject implements Subject /* * 你好 * * param name * return */ public String SayHello(String name) return hello + name; /* * 再见 * * return */ public String SayGoodBye return good bye ; 3、调用处理器实现类 java view plain copy 在CODE上查看代码片派生到我的代码片 package jiankunking; import java.lang.refl
8、ect.InvocationHandler; import java.lang.reflect.Method; /* * 调用处理器实现类 * 每次生成动态代理类对象时都需要指定一个实现了该接口的调用处理器对象 */ public class InvocationHandlerImpl implements InvocationHandler /* * 这个就是我们要代理的真实对象 */ private Object subject; /* * 构造方法,给我们要代理的真实对象赋初值 * * param subject */ public InvocationHandlerImpl(Objec
9、t subject) this.subject = subject; /* * 该方法负责集中处理动态代理类上的所有方法调用。 * 调用处理器根据这三个参数进行预处理或分派到委托类实例上反射执行 * * param proxy 代理类实例 * param method 被调用的方法对象 * param args 调用参数 * return * throws Throwable */ public Object invoke(Object proxy, Method method, Object args) throws Throwable /在代理真实对象前我们可以添加一些自己的操作 Syst
10、em.out.println(在调用之前,我要干点啥呢?); System.out.println(Method: + method); /当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用 Object returnValue = method.invoke(subject, args); /在代理真实对象后我们也可以添加一些自己的操作 System.out.println(在调用之后,我要干点啥呢?); return returnValue; 4、测试 java view plain copy 在CODE上查看代码片派生到我的代码片
11、 package jiankunking; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Proxy; /* * 动态代理演示 */ public class DynamicProxyDemonstration public static void main(String args) /代理的真实对象 Subject realSubject = new RealSubject; /* * InvocationHandlerImpl 实现了 InvocationHandler 接口,并能实现方法调用从代理
12、类到委托类的分派转发 * 其内部通常包含指向委托类实例的引用,用于真正执行分派转发过来的方法调用. * 即:要代理哪个真实对象,就将该对象传进去,最后是通过该真实对象来调用其方法 */ InvocationHandler handler = new InvocationHandlerImpl(realSubject); ClassLoader loader = handler.getClass.getClassLoader; Class interfaces = realSubject.getClass.getInterfaces; /* * 该方法用于为指定类装载器、一组接口及调用处理器生成
13、动态代理类实例 */ Subject subject = (Subject) Proxy.newProxyInstance(loader, interfaces, handler); System.out.println(动态代理对象的类型:+subject.getClass.getName); String hello = subject.SayHello(jiankunking); System.out.println(hello); / String goodbye = subject.SayGoodBye; / System.out.println(goodbye); 5、输出结果如下
14、: 四、动态代理怎么实现的? 从使用代码中可以看出,关键点在: java view plain copy 在CODE上查看代码片派生到我的代码片 Subject subject = (Subject) Proxy.newProxyInstance(loader, interfaces, handler); 通过跟踪提示代码可以看出:当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用。 也就是说,当代码执行到: subject.SayHello(jiankunking)这句话时,会自动调用InvocationHandlerImpl的inv
15、oke方法。这是为啥呢? =横线之间的是代码跟分析的过程,不想看的朋友可以直接看结论= 以下代码来自:JDK1.8.0_92 既然生成代理对象是用的Proxy类的静态方newProxyInstance,那么我们就去它的源码里看一下它到底都做了些什么? java view plain copy 在CODE上查看代码片派生到我的代码片 public static Object newProxyInstance(ClassLoader loader, Class interfaces, InvocationHandler h) throws IllegalArgumentException /检查h
16、 不为空,否则抛异常 Objects.requireNonNull(h); final Class intfs = interfaces.clone; final SecurityManager sm = System.getSecurityManager; if (sm != null) checkProxyAccess(Reflection.getCallerClass, loader, intfs); /* * 获得与指定类装载器和一组接口相关的代理类类型对象 */ Class cl = getProxyClass0(loader, intfs); /* * 通过反射获取构造函数对象并生
17、成代理类实例 */ try if (sm != null) checkNewProxyPermission(Reflection.getCallerClass, cl); /获取代理对象的构造方法 final Constructor cons = cl.getConstructor(constructorParams); final InvocationHandler ih = h; if (!Modifier.isPublic(cl.getModifiers) AccessController.doPrivileged(new PrivilegedAction public Void run
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- Java JDK 动态代理使用及实现原理分析 动态 代理 使用 实现 原理 分析
链接地址:https://www.31ppt.com/p-3159499.html