Java动态代理学习

原创 2015年11月19日 16:24:04

刚在Java虚拟机书上看到了说Java的动态代理的实现机制,使我困惑很久的问题终于被解开,所以在这里记录一下.
先奉上测试代码(每个类或接口一个.java文件):

//等会儿会被代理的接口
//Java自带的动态代理生成的是代理接口的对象
public interface SayHello {
    void sayHello();
}

//上面接口的一个实现类
public class HelloImpl implements SayHello {

    public void sayHello() {
        System.out.println("hello");
    }

}

//会自动生成代理对象的动态代理类
public class DynamicProxy implements InvocationHandler {

    //代理的是这个类的接口
    Object originObj;

    public DynamicProxy(Object originObj) {
        this.originObj = originObj;
    }

    public Object getProxy() {
        return Proxy.newProxyInstance(originObj.getClass().getClassLoader(), originObj.getClass().getInterfaces(), this);
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("这句是代理打印的!");
        return method.invoke(originObj, args);
    }

}

//main方法
public class DynamicProxyTest {

    public static void main(String[] args) {
        System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
        //先生成一个原始实现了SayHello接口的实例,因为需要这个实例去生成代理实例
        HelloImpl originHello = new HelloImpl();
        //得到可以动态生成代理实例的对象
        DynamicProxy proxy = new DynamicProxy(originHello);
        //得到代理SayHello接口的实例对象
        SayHello proxyHello = (SayHello) proxy.getProxy();
        proxyHello.sayHello();
    }
}

运行结果是:

这句是代理打印的!
hello

添加System.getProperties().put(“sun.misc.ProxyGenerator.saveGeneratedFiles”, “true”);这一条语句的作用是为了保存动态生成的代理类的字节码文件.

这里的黑匣子只有一句:

Proxy.newProxyInstance(originObj.getClass().getClassLoader(), originObj.getClass().getInterfaces(), this);

这条方法调用会在运行时生成一个代理类,代理SayHello接口的所有方法.
要看使如何做到得,必须查看自动生成的代理类的源码,这里用反编译工具jd-gui将生成的字节码文件反编译,得到的代码是:

package com.sun.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
import top.geekgao.proxy.SayHello;

public final class $Proxy0
  extends Proxy
  implements SayHello
{
  private static Method m1;
  private static Method m3;
  private static Method m2;
  private static Method m0;

  public $Proxy0(InvocationHandler paramInvocationHandler)
  {
    super(paramInvocationHandler);
  }

  public final boolean equals(Object paramObject)
  {
    try
    {
      return ((Boolean)this.h.invoke(this, m1, new Object[] { paramObject })).booleanValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  //主要看这里,这里是代理类的核心
  public final void sayHello()
  {
    try
    {
        //this.h代表的就是Proxy.newProxyInstance()的最后一个参数,即上面main方法中的proxy对象
        //有了h就可以调用它的invoke()方法
       //m3代表的就是sayHello()方法,并且没有参数,所以是null
       //这样就可以调用每一个方法了
      this.h.invoke(this, m3, null);
      return;
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final String toString()
  {
    try
    {
      return (String)this.h.invoke(this, m2, null);
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  public final int hashCode()
  {
    try
    {
      return ((Integer)this.h.invoke(this, m0, null)).intValue();
    }
    catch (Error|RuntimeException localError)
    {
      throw localError;
    }
    catch (Throwable localThrowable)
    {
      throw new UndeclaredThrowableException(localThrowable);
    }
  }

  static
  {
    try
    {
      m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[] { Class.forName("java.lang.Object") });
      m3 = Class.forName("top.geekgao.proxy.SayHello").getMethod("sayHello", new Class[0]);
      m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
      m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
      return;
    }
    catch (NoSuchMethodException localNoSuchMethodException)
    {
      throw new NoSuchMethodError(localNoSuchMethodException.getMessage());
    }
    catch (ClassNotFoundException localClassNotFoundException)
    {
      throw new NoClassDefFoundError(localClassNotFoundException.getMessage());
    }
  }
}

书上说的是生成字节码文件是通过class文件规范去拼装,这个就先不理他了.

版权声明:欢迎转载,注明出处即可。

彻底理解Java动态代理

代理设计模式 定义:为其他对象提供一种代理以控制对这个对象的访问。 代理模式的结构如下图所示。 动态代理使用 java动态代理机制以巧妙的方式实现了代理模式的设计理念。 ...
  • qq_29945661
  • qq_29945661
  • 2016年11月21日 22:06
  • 808

JAVA动态代理学习

代理模式分为静态代理和动态代理,静态代理中的代理是由程序员自己编写的,也就是说在我们的代码运行前,代理类的class编译文件就已存在;而动态代理与其相反,代理类是在运行阶段由JVM帮忙生成。java也...
  • yq6073025
  • yq6073025
  • 2016年09月30日 11:25
  • 156

java动态代理学习

代理类和目标类都实现同一个接口,而代理类在构造器中new一个目标类,并在代理类的方法中执行构造函数中new出的目标类的方法。 /**  * 相亲接口  *  * @author zhengt...
  • waterfallcode
  • waterfallcode
  • 2013年10月29日 19:31
  • 43

Java动态代理(主要是对六个参数的理解)

import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflec...
  • nrsc272420199
  • nrsc272420199
  • 2017年08月04日 00:33
  • 112

JAVA动态代理用法与实现过程

jdk动态代理实现已即原理
  • qq1004642027
  • qq1004642027
  • 2016年08月02日 11:33
  • 2090

详解java动态代理机制以及使用场景(一)

说起java动态代理,在我刚开始学java时对这项技术也是十分困惑,明明可以直接调通的对象方法为什么还要使用动态代理?随着学习的不断深入和工作经验的积累,慢慢的体会并理解了java动态代理机制。昨天再...
  • u011784767
  • u011784767
  • 2017年10月19日 10:29
  • 1030

Java动态代理与静态代理的定义与区别??

JAVA的静态代理与动态代理比较 一、概念 代理模式是常用的Java 设计模式,它的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后...
  • whb123andy
  • whb123andy
  • 2014年02月26日 13:38
  • 587

Java动态代理示例代码

Proxy 提供用于创建动态代理类和实例的静态方法。简洁方法:Foo f = (Foo) Proxy.newProxyInstance(Foo.class.getClassLoader(),     ...
  • anerou
  • anerou
  • 2007年06月20日 11:35
  • 3744

好记性不如烂笔头48-java拦截器-JDK自带动态代理和CGLIB效率比较(3)

Java中自带的动态代理的类必须要实现一个接口,而且据说使用反射的效率也并不是很高。于是CGLIB就诞生了。 使用CGLib实现动态代理,完全不受代理类必须实现接口的限制,而且CGLib底层采用AS...
  • ffm83
  • ffm83
  • 2015年02月11日 10:04
  • 1159

Java动态代理、cglib动态代理

说动态代理,需要先清楚静态代理。所谓静态代理就是程序员提前实现好的代理类,编译后class文件是已经存在的。 实现原理,利用Java代理模式,由一个代理类持有委托类的实例,并实现委托类一样的接口,来...
  • catoop
  • catoop
  • 2016年02月24日 14:17
  • 2848
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java动态代理学习
举报原因:
原因补充:

(最多只允许输入30个字)