详解Java中的静态代理和动态代理


代理是一种设计模式

在代理模式(Proxy Pattern)中,一个类代表另一个类的功能。这种类型的设计模式属于结构型模式。在代理模式中,我们创建具有现有对象的对象,以便向外界提供功能接口。目的:为其他对象提供一种代理以控制对这个对象的访问。

类关系图:


静态代理

创建一个接口,然后创建被代理的类实现该接口并且实现该接口中的抽象方法。之后再创建一个代理类,同时使其也实现这个接口。在代理类中持有一个被代理对象的引用,而后在代理类方法中调用该对象的方法。

代码如下:

接口

public interface HelloInterface{

        void sayHello();

}

被代理类

public class Hello implements HelloInterface{

       public void sayHello() {

               System.out.println("Hello Kevin!");

       }

}

代理类

public class HelloProxy implements HelloInterface{

    private HelloInterface helloInterface=new Hello();

    public void sayHello() {

       System.out.println("Beforeinvoke sayHello" );

       helloInterface.sayHello(); //调用被代理类Hello中的sayHello方法

       System.out.println("Afterinvoke sayHello");

    }

}

代理类调用

被代理类被传递给了代理类HelloProxy,代理类在执行具体方法时通过所持用的被代理类完成调用。

public class ProxyTest {

       public static void main(String[] args) {

            HelloProxy helloProxy=new HelloProxy();

            helloProxy.sayHello();

       }

}

静态代理的本质:由程序员创建或工具生成代理类的源码,再编译代理类。所谓静态也就是在程序运行前就已经存在代理类的字节码文件,代理类和委托类的关系在运行前就确定了。 

动态代理

动态代理类的源码是在程序运行期间由JVM根据反射等机制动态的生成,所以不存在代理类的字节码文件。代理类和委托类的关系是在程序运行时确定。JDK中关于动态代理的重要api如下:

java.lang.reflect.Proxy  这是Java 动态代理机制生成的所有动态代理类的父类,它提供了一组静态方法来为一组接口动态地生成代理类及其对象。 最重要的方法是:

static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h) ,该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例

java.lang.reflect.InvocationHandler  这是调用处理器接口,定义了一个invoke 方法,用于集中处理在动态代理类对象上的方法调用,通常在该方法中实现对委托类的代理访问。每次生成动态代理类对象时都要指定一个对应的调用处理器对象。Object invoke(Object proxy, Method method, Object[] args)  该方法负责集中处理动态代理类上的所有方法调用。第一个参数既是代理类实例,第二个参数是被调用的方法对象 ,第三个方法是调用参数。调用处理器根据这三个参数进行预处理或分派到委托类实例上反射执行

java.lang.ClassLoader  这是类装载器类,负责将类的字节码装载到Java 虚拟机(JVM)中并为其定义类对象,然后该类才能被使用。Proxy静态方法生成动态代理类同样需要通过类装载器来进行装载才能使用,它与普通类的唯一区别就是其字节码是由JVM 在运行时动态生成的而非预先存在于任何一个.class 文件中。 每次生成动态代理类对象时都需要指定一个类装载器对象。我们来看一下动态代理的实例:

接口

public interface HelloInterface{

        void sayHello();

}

被代理类

public class Hello implements HelloInterface{

       public void sayHello() {

               System.out.println("Hello Kevin!");

       }

}

实现InvocationHandler接口,创建自己的调用处理器 

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

public class ProxyHandler implements InvocationHandler{

    private Object object;

    public ProxyHandler(Object object){

        this.object = object;

    }

       public Object invoke(Object proxy,Method method, Object[] args) throws Throwable {

              System.out.println("Before invoke "  + method.getName());

             method.invoke(object, args);

             System.out.println("After invoke" + method.getName());

             return null;

       }

}

测试类

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Proxy;

public class DynamicProxyTest{

        public static void main(String[] args) {

           HelloInterface hello = new Hello();

          //把hello实例传入动态代理处理器

           InvocationHandler handler = new ProxyHandler(hello);

           //生成动态代理类实例

      HelloInterface proxyHello = (HelloInterface)        Proxy.newProxyInstance(hello.getClass().getClassLoader(),hello.getClass().getInterfaces(), handler);

               proxyHello.sayHello();

           }

}

运行代码

Before invoke sayHello

Hello Kevin!

After invoke sayHello

我们可以看到,动态代理与静态代理相比较,最大的好处是接口中声明的所有方法都被转移到调用处理器一个集中的方法中处理(InvocationHandler.invoke),生成不同类的代理实例我们只需要在类DynamicProxyTest中处理即可;而静态代理需要代理多个类的时候,由于代理对象要实现与目标对象一致的接口,则会遇到下面的问题:

只维护一个代理类,由这个代理类实现多个接口,但是这样就导致代理类过于庞大;

新建多个代理类,每个目标对象对应一个代理类,但是这样会产生过多的代理类当接口;

需要增加、删除、修改方法的时候,目标对象与代理类都要同时修改,不易维护。

动态代理类也有小小的遗憾,那就是它只能为接口创建代理!如果想对没有实现接口的类创建代理则无能为力。为了解决这种情况,我们通常使用cglib技术,其在AOP(例如spring)和ORM(例如Hibernate)中有广泛的应用,在这里就不对cglib进行展开介绍了。

动态代理类的生成

我们再来看一个实例,修改类DynamicProxyTest,代码如下:

public static void main(String[] args) {

               System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles","true"); 

               HelloInterface hello = new Hello();

               InvocationHandler handler = new ProxyHandler(hello);

               HelloInterface proxyHello=(HelloInterface) Proxy.newProxyInstance(hello.getClass().getClassLoader(), hello.getClass().getInterfaces(),handler);

               proxyHello.sayHello();

               System.out.println(proxyHello.getClass().getName());

           }

运行结果

Before invoke sayHello

Hello Kevin!

After invoke sayHello

com.sun.proxy.$Proxy0

我们发现proxyHello的类型是.$Proxy0而不是HelloInterface。我们通过反编译来查看$Proxy0的源码,在工程的com.sun.proxy目录下。注意:必须添加下面的代码

System.setProperty("sun.misc.ProxyGenerator.saveGeneratedFiles","true");

$Proxy0就是由JDK创建的动态代理类,是在运行时创建生成的。动态代理类的格式是“$ProxyN”,其中 N 是一个逐一递增的阿拉伯数字,代表Proxy 类第N 次生成的动态代理类,并不是每次调用Proxy 的静态方法创建动态代理类都会使得N 值增加,原因是如果对同一组接口(包括接口排列的顺序相同)试图重复创建动态代理类,它会很聪明地返回先前已经创建好的代理类的类对象,而不会再尝试去创建一个全新的代理类,这样可以节省不必要的代码重复生成,提高了代理类的创建效率。$Proxy0源码如下:

import com.my.demo2.HelloInterface;

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

import java.lang.reflect.Proxy;

import java.lang.reflect.UndeclaredThrowableException;

public final class $Proxy0 extends Proxy implements HelloInterface {

  private static Method m1;

  private static Method m3;

  private static Method m2;

  private static Method m0;

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

  public final boolean equals(ObjectparamObject) {

    try {

      return ((Boolean)this.h.invoke(this,m1, new Object[] { paramObject })).booleanValue();

    } catch (Error|RuntimeExceptionerror) {

      throw null;

    } catch (Throwable throwable) {

      throw newUndeclaredThrowableException(throwable);

    }

  }

  public final void sayHello() {

    try {

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

      return;

    } catch (Error|RuntimeExceptionerror) {

      throw null;

    } catch (Throwable throwable) {

      throw newUndeclaredThrowableException(throwable);

    }

  }

  public final String toString() {

    try {

      return (String)this.h.invoke(this,m2, null);

    } catch (Error|RuntimeExceptionerror) {

      throw null;

    } catch (Throwable throwable) {

      throw newUndeclaredThrowableException(throwable);

    }

  }

  public final int hashCode() {

    try {

      return ((Integer)this.h.invoke(this,m0, null)).intValue();

    } catch (Error|RuntimeExceptionerror) {

      throw null;

    } catch (Throwable throwable) {

      throw newUndeclaredThrowableException(throwable);

    }

  }

  static {

    try {

      m1 = Class.forName("java.lang.Object").getMethod("equals",new Class[] { Class.forName("java.lang.Object") });

      m3 = Class.forName("com.my.demo2.HelloInterface").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 (NoSuchMethodExceptionnoSuchMethodException) {

      throw newNoSuchMethodError(noSuchMethodException.getMessage());

    } catch (ClassNotFoundExceptionclassNotFoundException) {

      throw new NoClassDefFoundError(classNotFoundException.getMessage());

    }

  }

}

从上面的代码中我们可以看到:

1.  在代理类$ProxyN的实例上调用其代理的接口中所声明的方法时,这些方法最终都会由调用处理器的invoke 方法执行;

2.     代理类的根类java.lang.Object 中的三个方法:hashCode,equals 和 toString也同样会被分派到调用处理器的invoke 方法中执行。

静态代理和动态代理最重要的四个知识点

1.静态代理在程序运行前就已经存在代理类的字节码文件中确认了代理类和委托类的关系;

2.动态代理类的源码是在程序运行期间由JVM根据反射等机制动态的生成,所以不存在代理类的字节码文件。代理类和委托类的关系是在程序运行时确定。 动态代理根据接口或目标对象,计算出代理类的字节码,然后再加载到JVM中使用。其实现原理如下:由于JVM通过字节码的二进制信息加载类的,那么,如果我们在运行期系统中,遵循Java编译系统组织.class文件的格式和结构,生成相应的二进制数据,然后再把这个二进制数据加载转换成对应的类,这样,就完成了在代码中,动态创建一个类的能力了。

3.静态代理的缺点是在程序规模稍大时,维护代理类的成本高,静态代理无法胜任;

4.动态代理只能为实现了接口的类创建代理。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容

  • Subject类,定义了RealSubject和Proxy的共用接口,这样就在任何使用RealSubject的地方...
    联旺阅读 214评论 0 0
  • 一、代理概念 为某个对象提供一个代理,以控制对这个对象的访问。 代理类和委托类有共同的父类或父接口,这样在任何使用...
    wyatt_plus阅读 780评论 0 5
  • 1、代理概念 为某个对象提供一个代理,以控制对这个对象的访问。 代理类和委托类有共同的父类或父接口,这样在任何使用...
    孔垂云阅读 7,309评论 4 54
  • 最近常常看到一些资料时,是不是会看到动态代理,但是在项目中却好像没怎么使用过动态代理,所以对动态代理的理解...
    Android开发_Hua阅读 2,511评论 0 2
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,471评论 28 53