Cglib和jdk动态代理的区别

动态代理解决了方法之间的紧耦合,

IOC解决了类与类之间的紧耦合!

Cglib和jdk动态代理的区别?

1、Jdk动态代理:利用拦截器(必须实现InvocationHandler)加上反射机制生成一个代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理

2、 Cglib动态代理:利用ASM框架,对代理对象类生成的class文件加载进来,通过修改其字节码生成子类来处理

什么时候用cglib什么时候用jdk动态代理?

1、目标对象生成了接口 默认用JDK动态代理

2、如果目标对象使用了接口,可以强制使用cglib

3、如果目标对象没有实现接口,必须采用cglib库,Spring会自动在JDK动态代理和cglib之间转换

JDK动态代理和cglib字节码生成的区别?

1、JDK动态代理只能对实现了接口的类生成代理,而不能针对类

2、Cglib是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法,并覆盖其中方法的增强,但是因为采用的是继承,所以该类或方法最好不要生成final,对于final类或方法,是无法继承的

Cglib比JDK快?

1、cglib底层是ASM字节码生成框架,但是字节码技术生成代理类,在JDL1.6之前比使用java反射的效率要高

2、在jdk6之后逐步对JDK动态代理进行了优化,在调用次数比较少时效率高于cglib代理效率

3、只有在大量调用的时候cglib的效率高,但是在1.8的时候JDK的效率已高于cglib

4、Cglib不能对声明final的方法进行代理,因为cglib是动态生成代理对象,final关键字修饰的类不可变只能被引用不能被修改

Spring如何选择是用JDK还是cglib?

1、当bean实现接口时,会用JDK代理模式

2、当bean没有实现接口,用cglib实现

3、可以强制使用cglib(在spring配置中加入<aop:aspectj-autoproxy proxyt-target-class=”true”/>)

一. Cglib原理

动态生成一个要代理的子类,子类重写要代理的类的所有不是final的方法。在子类中采用方法拦截技术拦截所有的父类方法的调用,顺势织入横切逻辑,它比Java反射的jdk动态代理要快

Cglib是一个强大的、高性能的代码生成包,它被广泛应用在许多AOP框架中,为他们提供方法的拦截

最底层的是字节码Bytecode,字节码是java为了保证依次运行,可以跨平台使用的一种虚拟指令格式

在字节码文件之上的是ASM,只是一种直接操作字节码的框架,应用ASM需要对Java字节码、class结构比较熟悉

位于ASM上面的是Cglib,groovy、beanshell,后来那个种并不是Java体系中的内容是脚本语言,他们通过ASM框架生成字节码变相执行Java代码,在JVM中程序执行不一定非要写java代码,只要能生成java字节码,jvm并不关系字节码的来源

位于cglib、groovy、beanshell之上的就是hibernate和spring AOP

最上面的是applications,既具体应用,一般是一个web项目或者本地跑一个程序、

使用cglib代码对类做代理?

使用cglib定义不同的拦截策略?

构造函数不拦截方法

用MethodInterceptor和Enhancer实现一个动态代理

Jdk中的动态代理

JDK中的动态代理是通过反射类Proxy以及InvocationHandler回调接口实现的,但是JDK中所有要进行动态代理的类必须要实现一个接口,也就是说只能对该类所实现接口中定义的方法进行代理,这在实际编程中有一定的局限性,而且使用反射的效率也不高

Cglib实现

使用cglib是实现动态代理,不受代理类必须实现接口的限制,因为cglib底层是用ASM框架,使用字节码技术生成代理类,你使用Java反射的效率要高,cglib不能对声明final的方法进行代理,因为cglib原理是动态生成被代理类的子类

Cglib的第三方库提供的动态代理

 1 /**
 2  * 动态代理:
 3  *  特点:字节码随用随创建,随用随加载
 4  *  作用:不修改源码的基础上对方法增强
 5  *  分类:
 6  *      基于接口的动态代理
 7  *      基于子类的动态代理
 8  *  基于子类的动态代理:
 9  *      涉及的类:Enhancer
10  *      提供者:第三方cglib库
11  *  如何创建代理对象:
12  *      使用Enhancer类中的create方法
13  *  创建代理对象的要求:
14  *      被代理类不能是最终类
15  *  newProxyInstance方法的参数:在使用代理时需要转换成指定的对象
16  *      ClassLoader:类加载器
17  *          他是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。固定写法
18  *      Callback:用于提供增强的代码
19  *          他是让我们写如何代理。我们一般写一个该接口的实现类,通常情况加都是匿名内部类,但不是必须的。
20  *          此接口的实现类,是谁用谁写。
21  *          我们一般写的都是该接口的子接口实现类,MethodInterceptor
22  */
23 com.dynamic.cglib.Producer cglibProducer= (com.dynamic.cglib.Producer) Enhancer.create(
24         com.dynamic.cglib.Producer.class,
25         new MethodInterceptor() {
26             /**
27              *  执行被代理对象的任何方法都会经过该方法
28              * @param obj
29              * @param method
30              * @param args
31              *      以上三个参数和基于接口的动态代理中invoke方法的参数是一样的
32              * @param proxy:当前执行方法的代理对象
33              * @return
34              * @throws Throwable
35              */
36             @Override
37             public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
38                 Object returnValue=null;
39                 Float money=(Float)args[0];
40                 if("saleProduct".equals(method.getName())){
41                    returnValue= method.invoke(producer,money*0.8f);
42                 }
43                 return returnValue;
44             }
45         }
46 );
47 cglibProducer.saleProduct(100.0f);

JDK本身提供的动态代理实现

 1 /**
 2          * 动态代理:
 3          *  特点:字节码随用随创建,随用随加载
 4          *  作用:不修改源码的基础上对方法增强
 5          *  分类:
 6          *      基于接口的动态代理
 7          *      基于子类的动态代理
 8          *  基于接口的动态代理:
 9          *      涉及的类:proxy
10          *      提供者:Jdk官方
11          *  如何创建代理对象:
12          *      使用Proxy类中的newProxyInstance方法
13          *  创建代理对象的要求:
14          *      被代理类最少实现一个接口,如果没有则不能使用
15          *  newProxyInstance方法的参数:在使用代理时需要转换成指定的对象
16          *      ClassLoader:类加载器
17          *          他是用于加载代理对象字节码的。和被代理对象使用相同的类加载器。固定写法
18          *      Class[]:字节码数组
19          *          它是用于让代理对象和被代理对象有相同方法。固定写法
20          *      InvocationHandler:用于提供增强的代码
21          *          他是让我们写如何代理。我们一般写一个该接口的实现类,通常情况加都是匿名内部类,但不是必须的。
22          *          此接口的实现类,是谁用谁写。
23          */
24        IProducer proxyProducer=  (IProducer) Proxy.newProxyInstance(
25                 producer.getClass().getClassLoader(),
26                 producer.getClass().getInterfaces(),
27
28                new InvocationHandler() {
29                    /**
30                     * 作用:执行被代理对象的任何接口方法都会经过该方法
31                     * @param proxy  代理对象的引用
32                     * @param method 当前执行的方法
33                     * @param args   当前执行方法所需的参数
34                     * @return       和被代理对象有相同返回值
35                     * @throws Throwable
36                     */
37                     @Override
38                     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
39 //                        提供增强的代码
40 //                        1、获取方法执行的参数
41                         Object returnValue=null;
42                         Float money=(Float)args[0];
43                         if("saleProduct".equals(method.getName())){
44                            returnValue= method.invoke(producer,money*0.8f);
45                         }
46                         return returnValue;
47                     }
48                 }
49         );

JDK和Cglib的区别:

 
Cglib

JDK

是否提供子类代理



是否提供接口代理

是(可强制)


区别

必须依赖于CGLib的类库,但是它需要类来实现任何接口代理的是指定的类生成一个子类,覆盖其中的方法

实现InvocationHandler

使用Proxy.newProxyInstance产生代理对象

被代理的对象必须要实现接口

引用:https://blog.csdn.net/yhl_jxy/article/details/80635012

原文地址:https://www.cnblogs.com/sandaman2019/p/12636727.html

时间: 2024-08-29 02:39:14

Cglib和jdk动态代理的区别的相关文章

CGlib和JDK动态代理

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

spring源码学习【准备】之jdk动态代理和cglib动态代理的区别和性能

一:区别:---->JDK的动态代理依靠接口实现,如果有些类并没有实现接口,则不能使用JDK代理,这就要使用cglib动态代理了.--->JDK的动态代理机制只能代理实现了接口的类,而不能实现接口的类就不能实现JDK的动态代理,cglib是针对类来实现代理的,他的原理是对指定的目标类生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理. 性能:--->jdk的动态代理由于jdk版本的升级,渐渐超越cglib 二:都说 Cglib 创建的动态代理的

zbb20180930 代理模式 -静态代理-jdk动态代理-cglib动态代理

CGLIB与JDK动态代理区别 区别: java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理.而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理. 1.如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP 2.如果目标对象实现了接口,可以强制使用CGLIB实现AOP 3.如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB

jdk动态代理和cglib动态代理的区别

CGLIB代理: CGLIB底层:使用字节码处理框架ASM,来转换字节码并生成新的类. CGLIB(CODE GENERLIZE LIBRARY)代理是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的所有方法,所以该类或方法不能声明称final的. 如果目标对象实现了接口,可以强制使用CGLIB实现代理. 如果目标对象没有实现接口,则默认会采用CGLIB代理: 为什么jdk动态代理必须基于接口 原因如下: 1.生成的代理类继承了Proxy,由于java是单继承,所以只能实现接口,通过接口

JDK动态代理与CGLIB动态代理区别

JDK动态代理 只能代理实现了接口的类 没有实现接口的类不能实现JDK的动态代理 CGLIB动态代理 针对类来实现代理的 对指定目标类产生一个子类,通过方法拦击技术拦截所有父类方法的调用 使用Cglib实现动态代理 package com.imooc.cglib; public class Train {  public void move(){   System.out.println("火车行驶中");  } } import java.lang.reflect.Method; i

Spring AOP JDK动态代理与CGLib动态代理区别

静态代理与动态代理 静态代理 代理模式 (1)代理模式是常用设计模式的一种,我们在软件设计时常用的代理一般是指静态代理,也就是在代码中显式指定的代理. (2)静态代理由 业务实现类.业务代理类 两部分组成.业务实现类 负责实现主要的业务方法,业务代理类负责对调用的业务方法作拦截.过滤.预处理,主要是在方法中首先进行预处理动作,然后调用业务实现类的方法 例子 /** * 定义一个账户接口 * @author Administrator */ public interface Count { //

DK动态代理与CGLib动态代理的区别

参考:JDK动态代理与CGLib动态代理的区别对比 1).JDK和CGLib的区别 JDK动态代理只能对实现了接口的类生成代理,而不能针对类 CGLib是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法(继承) 2).Spring在选择用JDK还是CGLib的依据 当Bean实现接口时,Spring就会用JDK的动态代理 当Bean没有实现接口时,Spring使用CGLib来实现 可以强制使用CGLib(在Spring配置中加入<aop:aspectj-autoproxy proxy

Java进阶之 JDK动态代理与Cglib动态代理

一.动态代理概述: 与静态代理对照(关于静态代理的介绍 可以阅读上一篇:JAVA设计模式之 代理模式[Proxy Pattern]), 动态代理类的字节码是在程序运行时由Java反射机制动态生成. 注意: 1.AspectJ是采用编译时生成AOP代理类,具有更好的性能,但是需要使用特定的编译器进行处理 2.Spring AOP采用运行时生成AOP代理类,无需使用特定编译器进行处理,但是性能相对于AspectJ较差 二.JDK动态代理 [对有实现接口的对象做代理] 1.JDK动态代理中 需要了解的

Java学习之:JDK动态代理与CGLIB动态代理

代理的概念:简单的理解就是通过为某一个对象创建一个代理对象,我们不直接引用原本的对象,而是由创建的代理对象来控制对原对象的引用. 动态代理:是指在程序运行时由Java反射机制动态生成,无需手动编写代码.动态代理不仅简化了编程工作,而且提高了软件系统的可扩展性,因为Java反射机制可以生成任意类型的动态代理类. 代理原理:代理对象内部含有对真实对象的引用,从而可以操作真实对象,同时代理对象提供与真实对象相同的接口以便在任何时刻都能代替真实对象.同时,代理对象可以在执行真实对象操作时,附加其他的操作