jdk的动态代理
1. java动态代理和cglib动态代理的区别
1、Jdk动态代理实例:复JDK动态代制理只能代理实现了接口的类,其他普通类不能实现。代理类会在newProxyInstance方法中生成
2、cglib动态代理:cglib是针对类实现代理的,为代理的类生成一个子类,覆盖方法实现增强,因为采用的是继承所以不能代理final修饰的类。需要cglib和asm两个jar包
2. jdk 动态代理与spring 动态代理区别
spring有两种动态代理方式,一种就是jdk动态代理,还有就是cglib动态代理,jdk动态代理只能代理接口,cglib可以是类。jdk代理生成速度比cglib快,性能没有cglib好
3. jdk动态代理与cglib动态代理有什么区别
1.JDK动态代理
此时代理对象和目标对象实现了相同的接口,目标对象作为代理对象的一个属性,具体接口实现中,可以在调用目标对象相应方法前后加上其他业务处理逻辑。
代理模式在实际使用时需要指定具体的目标对象,如果为每个类都添加一个代理类的话,会导致类很多,同时如果不知道具体类的话,怎样实现代理模式呢?这就引出动态代理。
JDK动态代理只能针对实现了接口的类生成代理。
2.CGLIB代理
CGLIB(CODE GENERLIZE LIBRARY)代理是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的所有方法,所以该类或方法不能声明称final的。
如果目标对象没有实现接口,则默认会采用CGLIB代理;
如果目标对象实现了接口,可以强制使用CGLIB实现代理(添加CGLIB库,并在spring配置中加入<aop:aspectj-autoproxy proxy-target-class="true"/>)。
AOP包括切面(aspect)、通知(advice)、连接点(joinpoint),实现方式就是通过对目标对象的代理在连接点前后加入通知,完成统一的切面操作。
4. 静态代理,JDK动态代理和CGLib动态代理之前的区别
1、静态代理:静态代理中的代理类,需要我们自己写
JDK动态代理类实现了InvocationHandler接口。在重写的invoke方法中可以看出,动态代理的基础是反射(method.invoke(对象,参数)),还好反射看的比较多,到现在还记得。在这里需要提到的是Proxy.newProxyInstance(),这个方法。字面上的意思是 新建一个代理类的实例,这一点就和静态代理不同了。里面的参数有三个 类加载器、所有的接口,得到InvocationHandler接口的子类实例。这就是JDK动态代理,该代理有以下几种特点:
1、Interface:对于JDK Proxy,业务类是需要一个Interface的,这是一个缺陷;
2、Proxy:Proxy类是动态产生的,这个类在调用Proxy.newProxyInstance()方法之后,产生一个Proxy类的实力。实际上,这个Proxy类也是存在的,不仅仅是类的实例,这个Proxy类可以保存在硬盘上;
3、Method:对于业务委托类的每个方法,现在Proxy类里面都不用静态显示出来
4、InvocationHandler:这个类在业务委托类执行时,会先调用invoke方法。invoke方法在执行想要的代理操作,可以实现对业务方法的再包装。
以上就是JDK动态代理
3、CGLib动态代理:上面的JDK Proxy只能代理实现了接口的类,而不能实现接口的类就不能实现JDK代理。这时候就需要CGLib动态代理类
这里需要注意的是实现MethodIntercetor接口,必须导入cglib-nodep-2.1_3.jar这个包。CGLib是针对类来实现代理的,他的原理是对指定的目标生成一个子类,并覆盖其中方法实现增强,但因为采用的是继承,所以不能对final修饰的类进行代理。
5. spring对AOP的支持 JDK动态代理和CGLIB的区别
Aspect默认情况下不来用实现接口,但对源于目标对象(UserManagerImpl.java),在默认情况下必须实现接口
如果没有实现接口必须引入CGLIB库
我们可以通过Advice中添加一个JoinPoint参数,这个值会由spring自动传入,从JoinPoint中可以取得
参数值、方法名等等
1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP
3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换
如何强制使用CGLIB实现AOP?
* 添加CGLIB库,SPRING_HOME/cglib/*.jar
* 在spring配置文件中加入<aop:aspectj-autoproxy proxy-target-class="true"/>
JDK动态代理和CGLIB字节码生成的区别?
* JDK动态代理只能对实现了接口的类生成代理,而不能针对类
* CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
因为是继承,所以该类或方法最好不要声明成final
6. JDK动态代理和CGLIB代理的区别
区别:
java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法回前调用InvokeHandler来处理。而cglib动态答代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP
3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换
7. JDK的动态代理为什么必须要使用接口与使用C
代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问。在内某些情况下,一个对象不适合或容者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。
接口是一种规范,定义了一组相似的行为。
通俗一些就是,当调用代理类的方法时与调用被代理类的方法时在写法上是没有任何区别的,只有接口才能保证这种一致性。
8. 如何获得jdk动态代理proxy的原始类
可以在通知中(如@Before等)通过JoinPoint获取class对象(proxy):point.getTarget().getClass();
如果proxy实现接口版,再通过class.getInterfaces()即可获取原始类;
如果proxy是使用继承权父类的方式,通过class.getSuperClass()即可获取原始类。
9. JDK动态代理生成的代理类在哪儿
spring代理类有用jdk的动态代理,也有用cglib包,cglib底层依赖asm包,asm同样在hibernate中也被使用,使用asm增强字节码,自动生成代理类,方法跟目标类基本一样
10. cglib动态代理和jdk动态代理的区别与应用
jdk动态代理:需要有顶层接口才能使用,但是在只有顶层接口的时候也可以使用,常版见是权mybatis的mapper文件是代理.
cglib动态代理:可以直接代理类,使用字节码技术,不能对 final类进行继承。使用了动态生成字节码技术。