CGLIB与JDK动态代理区别
jdk动态代理是由Java内部的反射机制来实现的,cglib动态代理底层则是借助asm来实现的。总的来说,反射机制在生成类的过程中比较高效,而asm在生成类之后的相关执行过程中比较高效(可以通过将asm生成的类进行缓存,这样解决asm生成类过程低效问题)。还有一点必须注意:jdk动态代理的应用前提,必须是目标类基于统一的接口。如果没有上述前提,jdk动态代理不能应用。
注:asm其实就是java字节码控制.
使用cglib[Code Generation Library]实现动态代理,并不要求委托类必须实现接口,底层采用asm字节码
生成框架生成代理类的字节码
原理:利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
实现代码如下(前提先引入 asm和cglib 的jar包)
package com.learn.cglib;
import java.lang.reflect.Method;
import com.learn.staticproxy.House;
import com.learn.staticproxy.YMK;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class CglibProxy implements MethodInterceptor {
@Override
public Object intercept(Object object, Method method, Object[] arr, MethodProxy methodProxy) throws Throwable {
System.out.println("中介开始监听...");
methodProxy.invokeSuper(object, arr);
System.out.println("中介结束监听...");
return null;
}
public static void main(String[] args) {
CglibProxy cglibProxy = new CglibProxy();
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(YMK.class);
enhancer.setCallback(cglibProxy);
House create = (House) enhancer.create();
create.sell();
}
}
package com.learn.staticproxy;
public interface House {
public void sell();
}
package com.learn.staticproxy;
public class YMK implements House {
@Override
public void sell() {
System.out.println("ymk要卖房...");
}
}
Cglib动态代理与JDK动态区别
java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。
而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
SpringAOP:
1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP
3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换
JDK动态代理只能对实现了接口的类生成代理,而不能针对类 。
CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法 。
因为是继承,所以该类或方法最好不要声明成final ,final可以阻止继承和多态。