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动态代理学习测试类

  • 2017年04月22日 20:45
  • 4KB
  • 下载

java动态代理学习

前几天看到java的动态代理机制,不知道是啥玩意,然后看了看。死活不知道 invoke(Object proxy, Method m, Object[] args)种的proxy是个什么东西,放在这...

java 之 动态代理学习示例

import java.lang.reflect.Method; public interface Advice { public void beforeMethod(Method method...
  • EthanQ
  • EthanQ
  • 2012年02月26日 17:05
  • 486

java 动态代理学习(Proxy,InvocationHandler)

原帖地址:点击打开链接 前几天看到java的动态代理机制,不知道是啥玩意,然后看了看。死活不知道  invoke(Object proxy, Method m, Object[] args)种的p...

java学习笔记13--反射机制与动态代理

本文地址:http://www.cnblogs.com/archimedes/p/java-study-note13.html,转载请注明源地址。 Java的反射机制 在Java运行时环境...

java动态代理的学习

---------------------- android培训、java培训、期待与您交流! ---------------------- 一、生活了中的代理   我们在生活中买东西大部分都是从...

java学习笔记---类型信息(type information)、反射机制与动态代理

(1)Class对象        要理解RTTI在java 中的工作原理,首先必须知道类型信息在运行时是如何表示的,这项工作是由成为Class对象的特殊对象完成的,它包含了与类有关的信息。 Cl...

Java动态代理学习文章(一)

引言 Java 动态代理机制的出现,使得 Java 开发人员不用手工编写代理类,只要简单地指定一组接口及委托类对象,便能动态地获得代理类。代理类会负责将所有的方法调用分派到委托对象上反射执行,在...

黑马程序员_JAVA动态代理技术学习笔记

---------------------- ASP.Net+Unity开发、.Net培训、期待与您交流! ---------------------- 1、代理的作用 1)要为已存在的,多个具...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Java动态代理学习
举报原因:
原因补充:

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