java 静态代理 JDK动态代理 Cglib动态代理

下面以一个简单的银行账户为例讲述讲述动态代理。

设计一个银行账户类,包含用户的账户余额,实现查询和更新余额功能

这个系统用了一段时间,有客户要求对账说账户余额给弄错了?因为上面没有存取款记录,最后银行不认账,客户收到了损失。银行为了避免这种现象再次发生,决定对这个系统进行修改,但是因为bankAccount太过复杂,希望在不修改bankAccount的情况下,增加日志功能。

静态代理

使用静态代理解决上面的问题。

银行要求所有模块都需要添加日志功能,这对苦逼的程序员来说真的是一个不小的工作量啊,这需要写多少个自定义的代理类???还有没有时间可以愉快的玩耍了!

动态代理

Java JDK提供了一种动态代理实现机制,不用为每一个类自己手动去编写一个代理类,它可以帮你自动生成代理类。

下面讲一下,上面用到的几个关键方法和接口:

InvocationHandler接口:  public interface InvocationHandler {  public Object invoke(Object proxy,Method method,Object[] args) throws Throwable;  }  参数说明:  Object proxy:动态生成的代理类的实例

Method method:要调用的方法  Object[] args:方法调用时所需要的参数 

可以将InvocationHandler接口的子类想象成一个代理的最终操作类,替换掉ProxySubject。 

Proxy类:  Proxy类是专门完成代理的操作类,可以通过此类为一个或多个接口动态地生成实现类,此类提供了如下的操作方法:  public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,  InvocationHandler h) throws IllegalArgumentException  参数说明:  ClassLoader loader:类加载器  Class<?>[] interfaces:得到全部的接口  InvocationHandler h:得到InvocationHandler接口的子类实例 

Ps:类加载器  在Proxy类中的newProxyInstance()方法中需要一个ClassLoader类的实例,ClassLoader实际上对应的是类加载器,在Java中主要有一下三种类加载器;  Booststrap ClassLoader:此加载器采用C++编写,一般开发中是看不到的;  Extendsion ClassLoader:用来进行扩展类的加载,一般对应的是jre\lib\ext目录中的类;  AppClassLoader:(默认)加载classpath指定的类,是最常使用的是一种加载器。 

 

bind() 通过Proxy的newProxyInstance创建了一个代理类对象。 

通过JDKProxyFactory我们可以看出来,无论bind传递任何类,bind都会帮你生成一个代理类。就可以很快的满足银行每个模块都添加日志的要求。

Cglib动态代理

JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。 Cglib是第三方的实现。

JDK实现动态代理需要实现类通过接口定义业务方法,对于没有接口的类,如何实现动态代理呢,这就需要CGLib了。CGLib采用了非常底层的字节码技术,其原理是通过字节码技术为一个类创建子类,并在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。JDK动态代理与CGLib动态代理均是实现Spring AOP的基础。 

CGLib创建的动态代理对象性能比JDK创建的动态代理对象的性能高不少,但是CGLib在创建代理对象时所花费的时间却比JDK多得多,所以对于单例的对象,因为无需频繁创建对象,用CGLib合适,反之,使用JDK方式要更为合适一些。同时,由于CGLib由于是采用动态创建子类的方法,对于final方法,无法进行代理。

 

静态代理,JDK动态代理,Cglib动态代理测试类如下:

源码,文档,所需jar下载地址(微信公众号:程序员之路):http://pan.baidu.com/s/1nuvzfkP

 

posted @ 2016-03-10 17:57  RedAnts  阅读(367)  评论(0编辑  收藏  举报