一、CGlib动态代理

 
 

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

字节码技术:参考:http://note.youdao.com/noteshare?id=13453e8d815d3102938a02881b6f418f&sub=E56D1E6223FC4CA8BF072CD045301EFA

 

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

 
 

简单的实现举例:

这是一个需要被代理的类,也就是父类,通过字节码技术创建这个类的子类,实现动态代理。

[java] view plain copy

1.  public class SayHello {  

2.   public void say(){  

3.    System.out.println("hello everyone");  

4.   }  

5.  }  

该类实现了创建子类的方法与代理的方法。getProxy(SuperClass.class)方法通过入参即父类的字节码,通过扩展父类的class来创建代理对象。intercept()方法拦截所有目标类方法的调用,obj表示目标类的实例,method为目标类方法的反射对象,args为方法的动态入参,proxy为代理类实例。proxy.invokeSuper(obj, args)通过代理类调用父类中的方法。

[java] view plain copy

1.  public class CglibProxy implements MethodInterceptor{  

2.   private Enhancer enhancer = new Enhancer();  

3.   public Object getProxy(Class clazz){  

4.    //设置需要创建子类的类  

5.    enhancer.setSuperclass(clazz);  

6.    enhancer.setCallback(this);  

7.    //通过字节码技术动态创建子类实例  

8.    return enhancer.create();  

9.   }  

10.
 //实现MethodInterceptor接口方法  

11.
 public Object intercept(Object obj, Method method, Object[] args,  

12.
   MethodProxy proxy) throws Throwable {  

13.
  System.out.println("前置代理");  

14.
  //通过代理类调用父类中的方法  

15.
  Object result = proxy.invokeSuper(obj, args);  

16.
  System.out.println("后置代理");  

17.
  return result;  

18.
 }  

19.
}  

具体实现类:

[java] view plain copy

1.  public class DoCGLib {  

2.   public static void main(String[] args) {  

3.    CglibProxy proxy = new CglibProxy();  

4.    //通过生成子类的方式创建代理类  

5.    SayHello proxyImp = (SayHello)proxy.getProxy(SayHello.class);  

6.    proxyImp.say();  

7.   }  

8.  }  

输出结果:

[plain] view plain copy

1.  前置代理  

2.  hello everyone  

3.  后置代理  

    
 

 
 

 
 

 
 

二、JDK动态代理

 

使用动态代理的五大步骤

1.通过实现InvocationHandler接口来自定义自己的InvocationHandler;

 
 

2.通过Proxy.getProxyClass获得动态代理类

 
 

3.通过反射机制获得代理类的构造方法,方法签名为getConstructor(InvocationHandler.class)

 
 

4.通过构造函数,获得代理对象,并将自定义的InvocationHandler实例对象传为参数传入

 
 

5.通过代理对象调用目标方法

 

 

public
class ObjectProxy implements
InvocationHandler {

 
 

    Object target = null;

    List<BeforeAdvice> beforeList = new ArrayList<BeforeAdvice>();

    List<AfterAdvice> afterList = new ArrayList<AfterAdvice>();

   
 

    public ObjectProxy() {

       super();

       // TODO Auto-generated constructor stub

    }

public ObjectProxy(Object target) {

        super();

       this.target = target;

     
 

       try{

//解析advice.xml中所有的前置通知--得到class的值--->通过反射得到类的对象

              DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();

              DocumentBuilder db = dbf.newDocumentBuilder();

              Document doc = db.parse("src//advice.xml");

             
 

              //遍历前置通知

              NodeList beforeNodeList = doc.getElementsByTagName("beforeAdvice");

              for(int i=0;i<beforeNodeList.getLength();i++){

                 Element beforeElement =(Element) beforeNodeList.item(i);

                 String classValue = beforeElement.getAttribute("class"); //类的完整路径

                 Class clz = Class.forName(classValue);

                 BeforeAdvice obj = (BeforeAdvice)clz.newInstance();

                 beforeList.add(obj);

              }

             
 

            //遍历后置通知

              NodeList afterNodeList = doc.getElementsByTagName("afterAdvice");

              for(int i=0;i<afterNodeList.getLength();i++){

                 Element afterElement =(Element) afterNodeList.item(i);

                 String classValue = afterElement.getAttribute("class"); //类的完整路径

                 Class clz = Class.forName(classValue);

                 AfterAdvice obj = (AfterAdvice)clz.newInstance();

                 afterList.add(obj);

              }

             
 

          
 

       }catch(Exception e){

         e.printStackTrace();

       }

    }

 
 

 
 

    /**

     * 伪装成目标类之后,要执行的方法

     */

    @Override

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

           throws Throwable {

       //短信验证

       for(BeforeAdvice bf:beforeList){

           //对method目标方法进行判断,执行响应的前置切面业务

           bf.Before();

       }

       //目标方法

       Object result = method.invoke(target, args);

       //日志记录

       for(AfterAdvice af:afterList){

           //对method目标方法进行判断,执行响应的后置切面业务

           af.After();

      
 

          
 

       }

       return result;

    }

   
 

    /**

     * 获得代理类的对象

     * @param obj

     * @return

     */

     public
static  Object getProxyBean(Object obj){    

       return Proxy.newProxyInstance(obj.getClass().getClassLoader(),

              obj.getClass().getInterfaces(),new ObjectProxy(obj));

   
 

    }

 
 

}

public
static
void main(String[] args) {

      Idog dog = (Idog)ObjectProxy.getProxyBean(new Dog());

      dog.say();

}

CGlib和JDK动态代理的更多相关文章

  1. 学习CGLIB与JDK动态代理的区别

    动态代理 代理模式是Java中常见的一种模式.代理又分为静态代理和动态代理.静态代理就是显式指定的代理,静态代理的优点是由程序员自行指定代理类并进行编译和运行,缺点是一个代理类只能对一个接口的实现类进 ...

  2. Cglib 与 JDK动态代理的运行性能比较

    都说 Cglib 创建的动态代理的运行性能比 JDK 动态代理能高出大概 10 倍,今日抱着怀疑精神验证了一下,发现情况有所不同,遂贴出实验结果,以供参考和讨论. 代码很简单,首先,定义一个 Test ...

  3. Cglib 与 JDK动态代理

    作者:xiaolyuh 时间:2019/09/20 09:58 AOP 代理的两种实现: jdk是代理接口,私有方法必然不会存在在接口里,所以就不会被拦截到: cglib是子类,private的方法照 ...

  4. 面试造火箭系列,栽在了cglib和jdk动态代理

    "喂,你好,我是XX巴巴公司的技术面试官,请问你是张小帅吗".声音是从电话那头传来的 "是的,你好".小帅暗喜,大厂终于找上我了. "下面我们来进行一 ...

  5. 输出cglib以及jdk动态代理产生的class文件

      --该设置用于输出jdk动态代理产生的类 System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles&q ...

  6. spring cglib 与 jdk 动态代理

    1. 概述 JDK动态代理是利用java反射机制 生成一个实现接口的匿名类, 在调用具体方法前调用InvocationHandler来处理 Cglib动态代理是 利用asm开源包 把被代理类的clas ...

  7. 有点深度的聊聊JDK动态代理

    在接触SpringAOP的时候,大家一定会被这神奇的功能所折服,想知道其中的奥秘,底层到底是如何实现的.于是,大家会通过搜索引擎,知道了一个陌生的名词:动态代理,慢慢的又知道了动态代理有多种实现方式, ...

  8. java学习笔记(中级篇)—JDK动态代理

    一.什么是代理模式 相信大家都知道代理商这个概念,在商业中,代理商无处不在.假设你要去买东西,你不可能去找真正的厂家去买,也不可能直接跟厂家提出需求,代理商就是这中间的一桥梁,连接买家和厂商.你要买或 ...

  9. AOP学习心得&jdk动态代理与cglib比较

    什么是AOP AOP(Aspect-OrientedProgramming,面向方面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善.OOP引入 ...

随机推荐

  1. bundle安装方法

    sudo chmod +x filename.bundle sudo ./filename .bundle 原文链接:http://www.chinastor.com/a/linux/ubuntu/0 ...

  2. ORACLE之常用FAQ V1.0

    [B]第一部分.SQL&PL/SQL[/B][Q]怎么样查询特殊字符,如通配符%与_[A]select * from table where name like 'A\_%' escape ' ...

  3. Mac终端Screen命令使用指南

    (1)创建会话 使用命令“screen -S RunWork”来创建一个screen会话,命令执行之后,就会得到一个新的shell窗口,为了便于标示可以用快捷键Ctrl-a A(就是按下Ctrl+a键 ...

  4. uva 610(tarjan的应用)

    题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=23727 思路:首先是Tarjan找桥,对于桥,只能是双向边,而对于 ...

  5. Pecan

    什么是peacn Pecan是一个轻量级的基于Python的Web框架, Pecan的目标并不是要成为一个“full stack”的框架, 因此Pecan本身不支持类似Session和Database ...

  6. JavaScript------创建节点

    代码: <script type="text/javascript"> var i = 0; function add(){ var inputNode = docum ...

  7. Hibernate_day04--HQL查询

    HQL查询 1 hql:hibernate query language,hibernate提供一种查询语言,hql语言和普通sql很相似,区别:普通sql操作数据库表和字段,hql操作实体类和属性 ...

  8. c#基础 第五讲

    using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...

  9. 【BZOJ1935/4822】[Shoi2007]Tree 园丁的烦恼/[Cqoi2017]老C的任务 树状数组

    题意:两道题差不多,都是给你一堆平面上的点,每个点有权值,然后m次询问求某一矩形区域内的点权和 题解:先离散化,然后将询问拆成左右两条线段,然后将点和这些线段一起按x坐标排序,在y轴上维护树状数组.然 ...

  10. 【POJ3565】ANTS KM算法

    [POJ3565]ANTS 题意:平面上有2*n个点,N白N黑.为每个白点找一个黑点与之连边,最后所有边不交叉.求一种方案. 题解:KM算法真是一个神奇的算法,虽然感觉KM能做的题用费用流都能做~ 本 ...