本文意在讲解CGLIB的基础使用及基本原理。
一、CGLIB的基本原理:
依赖ASM字节码工具,通过动态生成实现接口或继承类的类字节码,实现动态代理。
针对接口,生成实现接口的类,即implements方式;针对类,生成继承父类的类,即extends方式。
二、为什么使用CGLIB?
JDK的动态代理只能基于接口,有时候我们想基于类生成动态代理,这个时候CGLIB是一个选择。
没什么场景下是必须使用CGLIB生成类代理的(个人观点),如果有,可能是代码简洁,某些情况下性能较好。
CGLIB基于类生成动态代理需要注意?(CGLIB生成的代理是继承类的)
1. final声明的类是不能被代理的;
2. 类中的private,final方法不能被代理,static方法不生成代理方法。
二、使用方法:
基础示例代码:
public interface UserInterface { boolean login(int userid); } public class UserInterfaceImpl implements UserInterface { public boolean login(int userid) { System.out.println("do Login!"); return false; } }
1. 代理接口:
public class CGlibProxy implements net.sf.cglib.proxy.InvocationHandler{ //这里的InvocationHandler是cglib包中的 private UserInterface ref; public CGlibProxy(UserInterface ref){ this.ref = ref; } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("before"); method.invoke(ref, args); System.out.println("after"); return null; } } public class Test { public static void main(String[] args) throws IOException { Enhancer en = new Enhancer(); en.setSuperclass(UserInterface.class); en.setCallback(new CGlibProxy(new UserInterfaceImpl())); UserInterface interfaced = (UserInterface) en.create(); interfaced.login(1); } }
代理接口和JDK的使用方法基本没啥区别,传入被代理实例对象,调用实例的对象的method方法。
所以如果是代理接口,完全没必要使用CGLIB。
2. 代理类:
public class CGlibProxy implements MethodInterceptor{ public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("before"); Object o = proxy.invokeSuper(obj, args); System.out.println("after"); return o; } } public class Test { public static void main(String[] args) throws IOException { Enhancer en = new Enhancer(); en.setSuperclass(UserInterfaceImpl.class); en.setCallback(new CGlibProxy()); UserInterface interfaced = (UserInterface) en.create(); interfaced.login(1); } }
3. Callback接口
Callback即代理方法,上述示例中MethodInterceptor就是Callback的子接口。Callback定义一个空接口,可以方便扩展。
Enhancer中有两个设置Callback的方法: setCallback(Callback callback), setCallbacks(Callback[] callbacks)。
你可能定义多个Callback,然后定义不同的代理行为,如下:
public class LoginProxy implements MethodInterceptor{ public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("before login"); proxy.invokeSuper(obj, args); System.out.println("after loign"); return null; } } public class OtherProxy implements MethodInterceptor{ public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable { System.out.println("before"); proxy.invokeSuper(obj, args); System.out.println("after"); return null; } }
cglib代理只能调用一个代理方法,所以当设置多个Callback时,你还需要指定一个CallbackFilter,通过Method条件指定Callback,
CallbackFilter接口:
public interface CallbackFilter { int accept(Method method); //返回值为指定的Callback数组的下标索引 }
示例:
public class Test { public static void main(String[] args) throws IOException { Enhancer en = new Enhancer(); en.setSuperclass(UserInterfaceImpl.class); en.setCallbacks(new Callback[]{new LoginProxy(),new OtherProxy()}); //callback数组 en.setCallbackFilter(new CallbackFilter() { public int accept(Method method) { if(method.getName().equals("login")){ return 0; //索引为0 , 即 LoginProxy } return 1; // 索引为1, 即 OtherProxy } }); UserInterfaceImpl interfaced = (UserInterfaceImpl) en.create(); interfaced.login(1); interfaced.other(); } }
4. MethodInterceptor:MethodInterceptor接口是Callback的子接口,最常用。
public interface MethodInterceptorextends Callback {
//obj: 代理对象本身,即cglib生成的代理实例
//method: 被代理对象中的方法
//args:方法的参数
//proxy: 存储了代理类,也就是obj public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable; }
如何调用被代理的目标方法?
MethodProxy的 invokeSuper(obj,args) 方法;obj就是 代理对象本身 ,args是对象参数;
MethodProxy还有一个invoke(obj,args)方法;obj参数是被代理对象,没搞懂这个方法的意图,个人觉得没什么必要。
5. NamingPolicy,自定义代理类名称,默认实现是DefaultNamingPolicy
public interface NamingPolicy { String getClassName(String prefix, String source, Object key, Predicate names); }
6. GeneratorStrategy,字节码生成策略,默认实现是DefaultGeneratorStrategy
public interface GeneratorStrategy { byte[] generate(ClassGenerator cg) throws Exception; }
你可以重写DefaultGeneratorStrategy中的方法来替换字节码生成器,也可以访问或修改生成的字节码,如下:
Enhancer en = new Enhancer(); en.setStrategy(new DefaultGeneratorStrategy(){ protected byte[] transform(byte[] b) throws Exception { return b; //b 是生成的字节码 } protected ClassGenerator transform(ClassGenerator cg) throws Exception { return cg; //cg 是字节码生成器 } });
7. interceptDuringConstruction,设置构造函数中的方法调用是否使用代理方法,默认为true。
示例:
public class UserInterfaceImpl implements UserInterface { public UserInterfaceImpl(){ login(1); //构造函数中调用方法 } public boolean login(int userid) { System.out.println("do Login!"); return false; } } public class Test { public static void main(String[] args) throws IOException { Enhancer en = new Enhancer(); en.setInterceptDuringConstruction(true); en.setSuperclass(UserInterfaceImpl.class); en.setCallback(new LoginProxy()); UserInterfaceImpl interfaced = (UserInterfaceImpl) en.create(); interfaced.login(1); } }
运行测试代码,结果如下,即login方法被代理了两次。
en.setInterceptDuringConstruction(false) 时,运行结果如下,即login方法被代理了一次
8. 没有默认构造函数时创建代理的方法:
public class UserInterfaceImpl implements UserInterface { public UserInterfaceImpl(String param){ } public boolean login(int userid) { System.out.println("do Login!"); return false; } } public class Test { public static void main(String[] args) throws IOException { Enhancer en = new Enhancer(); en.setSuperclass(UserInterfaceImpl.class); en.setCallback(new LoginProxy()); UserInterfaceImpl interfaced = (UserInterfaceImpl) en.create(new Class[]{String.class}, new Object[]{"name"}); //创建方法中指定构造函数的参数类型及对应的参数值 interfaced.login(1); } }
三、cglib创建类代理的基本原理
如果让你实现类代理? ---- 难点在哪里?
1. 创建代理的过程:
根据各种参数生成缓存的key --> 生成代理类(先缓存获取,缓存没有则用ClassGenerator生成代理类存入缓存) --> 根据代理类构造器生成代理对象实例。
2. 代理类对象:
可以设置系统参数cglib.debugLocation,开启代理类存入文件,该参数为文件存储路径。
示例代码:
public class UserInterfaceImpl implements UserInterface { public boolean login(int userid) { return false; } } public class Test { public static void main(String[] args) throws IOException { System.setProperty("cglib.debugLocation", "E://test"); Enhancer en = new Enhancer(); en.setSuperclass(UserInterfaceImpl.class); en.setCallback(new LoginProxy()); en.create(); } }
运行后,test目录下会生成一些class文件,找到同包(自己的package)目录,反编译打开(用的luyten,比jdgui好用)
为了方便看,删除了一些不必要的代码;
package cglibproxy; import java.lang.reflect.*; import net.sf.cglib.proxy.*; import net.sf.cglib.core.*; public class UserInterfaceImpl$$EnhancerByCGLIB$$f2d3e293 extends UserInterfaceImpl implements Factory {private MethodInterceptor CGLIB$CALLBACK_0; private static Object CGLIB$CALLBACK_FILTER;
private static final Method CGLIB$login$0$Method; private static final MethodProxy CGLIB$login$0$Proxy;
private static final Method CGLIB$equals$1$Method; private static final MethodProxy CGLIB$equals$1$Proxy;
private static final Method CGLIB$toString$2$Method; private static final MethodProxy CGLIB$toString$2$Proxy;
private static final Method CGLIB$hashCode$3$Method; private static final MethodProxy CGLIB$hashCode$3$Proxy;
private static final Method CGLIB$clone$4$Method; private static final MethodProxy CGLIB$clone$4$Proxy; final boolean CGLIB$login$0(final int n) { return super.login(n); } public final boolean login(final int n) { MethodInterceptor cglib$CALLBACK_2; MethodInterceptor cglib$CALLBACK_0; if ((cglib$CALLBACK_0 = (cglib$CALLBACK_2 = this.CGLIB$CALLBACK_0)) == null) { CGLIB$BIND_CALLBACKS(this); cglib$CALLBACK_2 = (cglib$CALLBACK_0 = this.CGLIB$CALLBACK_0); } if (cglib$CALLBACK_0 != null) { final Object intercept = cglib$CALLBACK_2.intercept((Object)this, UserInterfaceImpl$$EnhancerByCGLIB$$f2d3e293.CGLIB$login$0$Method, new Object[] { new Integer(n) }, UserInterfaceImpl$$EnhancerByCGLIB$$f2d3e293.CGLIB$login$0$Proxy); return intercept != null && (boolean)intercept; } return super.login(n); } }
可以从反编译的代理类中看到:
代理类继承了被代理类,Factory接口是cglib的内部接口,有兴趣的可以去看一下;
代理类代理了两种方法,一种是被代理类(即我们自定义的方法),一种是Object中的4个方法(toString, equals, hashCode, clone);
代理类针对每个方法,生成了两个方法(一个代理方法,一个原方法) (为什么这么做?想想MethodProxy.invokeSuper(),这里是一个关键点);
代理类中的目标方法都使用了final声明,禁止继续被代理;
以上就是个人总结的cglib的一些基础,水平有限,有问题欢迎评论中指正,谢谢!
原创文章,转载请注明出处。