java 代理模式 CGLIB的动态代理

       JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,这样就存在一定的局限性。对于这种情况,我们采用CGLIB来实现。

一、CGLIB动态代理

     cglib是针对类来实现代理的,其实现原理:CGLIB的底层采用ASM字节码生成框架,使用字节码技术生成代理,比使用反射生成代理的效果要高,是对指定的目标类生成一个子类,并覆盖其中方法实现增强。但是也有一点点不足,因为采用的是继承,所以不能对final修饰的类进行代理。 

    还是使用以前的代码,依然还是简单的三步来实现。第一:建立一个普普通通的业务类;第二:写CGLIB代理类;第三:写测试代码或者客户端调用。这里的不同是第一步中,我们不需要在建接口了,只是一个普普通通的java类。

看代码:

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;"/**    
  2.      * 业务类(包含业务逻辑)    
  3.      *  即:委托类   
  4.      * @author Cassie     
  5.      */        
  6.     public class Account{        
  7.             
  8.         @Override        
  9.         public void queryAccount() {        
  10.             System.out.println("查询方法...");              
  11.         }        
  12.             
  13.         @Override        
  14.         public void updateAccount() {        
  15.             System.out.println("修改方法...");              
  16.         }        
  17.             
  18.     }</span> 
 

然后,我们来写CGLIB代理类:

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">import java.lang.reflect.Method;   
  2. import net.sf.cglib.proxy.Enhancer;    
  3. import net.sf.cglib.proxy.MethodInterceptor;    
  4. import net.sf.cglib.proxy.MethodProxy;    
  5.     
  6. /**  
  7.  * 使用cglib动态代理  
  8.  * @author Cassie 
  9.  * 实现MethodInterceptor 接口 
  10.  */    
  11. public class CglibProxy implements MethodInterceptor {    
  12.   
  13.     //委托对象,运行时定类型  
  14.     private Object target;    
  15.     
  16.     /**  
  17.      * 创建代理对象  
  18.      *   
  19.      * @param target  
  20.      * @return  
  21.      */    
  22.     public Object getInstance(Object target) {    
  23.         this.target = target;    
  24.         Enhancer enhancer = new Enhancer();    
  25.         enhancer.setSuperclass(this.target.getClass());    
  26.         // 回调方法    
  27.         enhancer.setCallback(this);    
  28.         // 创建代理对象    
  29.         return enhancer.create();    
  30.     }    
  31.     
  32.     @Override    
  33.     // 回调方法    

  34.     public Object intercept(Object obj, Method method, Object[] args,    
  35.             MethodProxy proxy) throws Throwable {    
  36.         System.out.println("before");    
  37.         Object result = proxy.invokeSuper(obj, args);    
  38.         System.out.println("after");    
  39.         return result  
  40.     }    
  41.     
  42. }  </span>  

最后写测试类调用

  1. <span style="font-family:KaiTi_GB2312;font-size:18px;">public class TestCglib {    
  2.         
  3.     public static void main(String[] args) {   
  4.     //实例化代理   
  5.         CglibProxy cglib=new CglibProxy();   
  6.     //通过代理拿到对象   
  7.         Account account = (Account)cglib.getInstance(new Account());    
  8.         account.query();    
  9.     }    
  10. }</span>  

            通过以上代码,我们发现proxy.invokeSuper(obj,arg)是执行的关键。

使用CGLIB,需要实现 CGLib 给我们提供的 MethodInterceptor 实现类,并填充 intercept() 方法。方法中最后一个 MethodProxy 类型的参数 proxy,值得注意!CGLib 给我们提供的是方法级别的代理,也可以理解为对方法的拦截。我们直接调用 proxy 的 invokeSuper() 方法,将被代理的对象 obj 以及方法参数 args 传入其中即可。

         至此,CGLIB代理也实现了。

转载自:http://blog.csdn.net/wangyongxia921/article/details/46240877

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值