1. CGLIB动态代理简介
JDK动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler
来处理。
而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
2. JDK动态代理与CGLIB区别
JDK动态代理和CGLIB字节码生成的区别?
- JDK动态代理只能对实现了接口的类生成代理,而不能针对类
- CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法。因为是继承,所以该类或方法最好不要声明成final
3.代码示例
注:需要引入相关jar包,下载cglib.jar和asm.jar
1. 真实对象
package dynamicproxy.cglib;
/**
* 真实对象
* @author <u>sunlh</u>
*
*/
public class HelloWorld {
public void sayHelloWorld() {
System.out.println("Hello World!");
}
}
2. 建立代理对象与真实服务对象的关系&实现代理逻辑方法
package dynamicproxy.cglib;
import java.lang.reflect.Method;
import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
public class CglibProxyExample implements MethodInterceptor{
/**
* 生成CGLIB代理对象
* @param cls Class类
* @return Class类的CGLIB代理对象
*/
public Object getProxy(<u>Class</u> cls) {
// CGLIB <u>enhancer</u>增强类对象
Enhancer enhancer = new Enhancer();
// 设置增强类型
enhancer.setSuperclass(cls);
// 定义代理逻辑对象为当前对象,要求当前对象实现MethodInterceptor方法
enhancer.setCallback(this);
// 生成并返回代理对象
return enhancer.create();
}
@Override
public Object intercept(Object proxy, Method method,
Object[] args, MethodProxy methodProxy) throws Throwable {
System.out.println("调用真实对象前");
// CGLIB反射调用真实对象方法
Object result = methodProxy.invokeSuper(proxy, args);
System.out.println("调用真实对象后");
return result;
}
}
这里使用CGLIB的加强者EnHancer
,通过设置超类的方法(setSuperclass
),然后通过setCallback
方法设置哪个类为它的代理类。其中,参数为this
就意味着是当前对象,那就要求用this
这个对象实现接口MethodInterceptor
的方法intercept,然后返回代理对象。
那么此时当前类的intercept方法就是其代理逻辑方法。
在反射真实对象前后进行打印,CGLIB通过如下方法完成
Object result = methodProxy.invokeSuper(proxy, args);
3. 测试CGLIB动态代理
package dynamicproxy.cglib;
/**
* 测试
* @author sunlh
*
*/
public class TestCglibProxy {
public static void main(String[] args) {
CglibProxyExample cpe = new CglibProxyExample();
HelloWorld proxy = (HelloWorld) cpe.getProxy(HelloWorld.class);
proxy.sayHelloWorld("张三");
}
}
控制台输出
调用真实对象前
Hello 张三
调用真实对象后
4.特殊情况
当真实对象声明为final
时,使用CGLIB动态代理会报错
代码示例
修改真实对象,声明为final(即不可继承)
package dynamicproxy.cglib;
/**
* 真实对象
* @author <u>sunlh</u>
*
*/
public final class HelloWorld {
public void sayHelloWorld(String name) {
System.out.println("Hello " + name);
}
}
再次执行测试方法,控制台输出
java.lang.IllegalArgumentException: Cannot subclass final class class dynamicproxy.cglib.HelloWorld