博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
动态代理技术的实现与理解
阅读量:4223 次
发布时间:2019-05-26

本文共 3772 字,大约阅读时间需要 12 分钟。

​ 最近在看《深入理解java虚拟机 –JVM高级特性与最佳实战》这本书,看到动态代理这个部分,虽然以前学习spring的时候就学过动态代理的知识,但是这次看到,突然感受到了不一样的理解,特此分享出来,请大家多多指教。

​ 动态代理实现小例子:

package ObjectCreate;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;public class DynamicProxyTest {
interface IHello{
void sayHello();}static class Hello implements IHello{
@Override public void sayHello() { System.out.println("hello World"); }}static class DynamicProxy implements InvocationHandler{
Object originalObj; Object bind(Object originalObj){ this.originalObj = originalObj; //通过该方法创建该对象的接口的代理对象的实例 //步骤2 Object obj = Proxy.newProxyInstance(originalObj.getClass().getClassLoader(), originalObj.getClass().getInterfaces(), this); return obj; } //步骤5 @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("welecome!"); //通过反射调用实例的方法 return method.invoke(originalObj, args); }}public static void main(String[] args) { //步骤1 IHello hello = (IHello) new DynamicProxy().bind(new Hello()); //步骤3 hello.sayHello();}}

执行结果:

这里写图片描述

分析:上述代码中,唯一的“黑匣子”就是Proxy.newProxyInstance()方法,除此之外再没有任何特殊之处。这个方法返回了一个实现IHello的接口,并且代理了new Hello()实例行为的对象。跟踪这个方法的源码,可以看到其在运行时动态的生成一个描述代理类的字节码byte[] 数组,在main方法中加入下面这段代码可以在磁盘中生成该代理类的class文件:$proxy().class。

System.getProperties().put(“sun.misc.ProxyGenerator.saveGeneratedFiles”,”true”);

将该类进行反编译得到以下代码:

package ObjectCreate;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.lang.reflect.UndeclaredThrowableException;public class $Proxy0 extends Proxy implements DynamicProxyTest.IHello{
private static Method m3;private static Method m1;private static Method m0;private static Method m2;protected $Proxy0(InvocationHandler h) { super(h);}@Overridepublic final void sayHello(){ try{ //步骤4 this.h.invoke(this,m3,null); return; }catch(RuntimeException localRuntimeException){ throw localRuntimeException; }catch(Throwable localThrowable){ throw new UndeclaredThrowableException(localThrowable); }}//此处由于版面原因,忽略equals()、hashCode()、toString()、三个方法的代码//这三个内容与sayHello()方法非常相似。static { try{ m3 = Class.forName("org.fenixsoft.bytecode.DynamicProxyTest$IHello").getMethod("sayHello", new Class[0]); m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")}); m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]); m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]); }catch(NoSuchMethodException localNoSuchMethodException){ throw new NoSuchMethodError(localNoSuchMethodException.getMessage()); }catch(ClassNotFoundException localClassNotFoundException){ throw new NoClassDefFoundError(localClassNotFoundException.getMessage()); }}}

以上全是书中写的,以下,为我自己的理解:

代码执行关键步骤:
1、IHello hello = (IHello) new DynamicProxy().bind(new Hello());
分析:该行代码将Hello的实例传递给DynamicProxy对象,返回IHello接口类型的对象($Proxy0这个类的实例)。

2、Object obj = Proxy.newProxyInstance(originalObj.getClass().getClassLoader(), originalObj.getClass().getInterfaces(), this);

分析:该代码在程序运行时动态的生成了$Proxy0这个类的字节码,如以上代码所示,这个类实现了IHello接口的sayHello()方法。
注意:这个方法返回值是$Proxy的实例对象,所以步骤1接受的是这个。

3、hello.sayHello();

分析:通过以上分析,我们可以很明确的理解此sayHello()方法应该是$Proxy的实例中的sayHello()方法。

4、this.h.invoke(this,m3,null);

分析:this表示当前方法所属实例对象、即$Proxy的实例,this.h是父类Proxy中保存的InvocationHandler实例变量也就是DynamicProxy 这个类的实例,事实上该段代码就是调用DynamicProxy 类的invoke方法,传递的参数为sayHello()方法。

5、public Object invoke(Object proxy, Method method, Object[] args)

分析:打印“welecome”语句,然后调用Hello实例的sayHello()方法,打印“hello World” 语句!分析完毕。

你可能感兴趣的文章
浅谈HashMap,HashTable,ConcurrentHashMap,WeakHashMap,HashMap源码分析
查看>>
云创大数据校企合作项目斩获“全国校企合作十佳案例”
查看>>
云创大数据与宽泛科技签订战略合作协议
查看>>
免费!免费!免费!全国高校大数据师资实战免费培训班
查看>>
“南京市独角兽瞪羚企业俱乐部创始人简餐会”走进云创
查看>>
本科大数据专业该怎么上?
查看>>
云创大数据1+X大数据应用部署与调优职业技能等级证书预申报正式开启!
查看>>
人工智能需要一个可被证明的理论作为基础 | 哈佛丘成桐
查看>>
入门 | 一文概览深度学习中的激活函数
查看>>
一分钟整明白Tensorflow Extended
查看>>
人工智能再次参加高考:和作家比写作文,AI能打多少分?
查看>>
云创冬日紫金山踏雪游记
查看>>
西安思源学院电子信息工程学院院长张卫钢一行到访
查看>>
邀请函|欢迎参加2019云创大数据实验平台金融类/电子商务类/数学统计类院校各省总代理招募大会!...
查看>>
云创大数据的2018年!
查看>>
QNX简介
查看>>
MQTT协议基本介绍
查看>>
进程和线程是操作系统基本概念,了解一下
查看>>
SSL与TLS的区别以及介绍
查看>>
HTTPS、TLS、SSL、HTTP区别和关系
查看>>