Cglib(Code Generation Library )是一个强大的,高性能,高质量的Code生成类库。它可以在运行期扩展Java类与实现Java接口。
Cglib 封装了asm,可以在运行期动态生成新的class。
Cglib 用于AOP,jdk中的proxy必须基于接口,cglib却没有这个限制。
原理区别:
java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
2、如果目标对象实现了接口,可以强制使用Cglib实现AOP
3、如果目标对象没有实现了接口,必须采用 Cglib 库,spring会自动在JDK动态代理和 Cglib 之间转换
JDK动态代理和CGLIB字节码生成的区别?
1、JDK动态代理只能对实现了接口的类生成代理,而不能针对类
2、 Cglib 是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
因为是继承,所以该类或方法最好不要声明成final
java动态代码例子:
package com.cglibjava.proxy;
public interface InfoDao {
void query();
}
package com.cglibjava.proxy;
public class InfoDaoImpl implements InfoDao{
public void query() {
System.out.println(" query ...... ");
}
}
package com.cglibjava.proxy;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
public class LogHandler implements InvocationHandler {
private Object target = null;
public LogHandler(Object target) {
this.target = target;
}
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println(method.getName() + "before invoke method");
Object result = method.invoke(target, args);
System.out.println(method.getName() + "after invoke method");
return result;
}
}
package com.cglibjava.proxy;
import java.lang.reflect.Proxy;
public class ProxyTest {
public static void main(String[] args) {
InfoDao infoDao = new InfoDaoImpl();
LogHandler logHandler = new LogHandler(infoDao);
InfoDao infoDaoProxy = (InfoDao) Proxy.newProxyInstance(infoDao
.getClass().getClassLoader(), infoDao.getClass()
.getInterfaces(), logHandler);
infoDaoProxy.query();
}
}