作者shensy及出处http://shensy.iteye.com/blog/1881130
实战CGLib系列文章
本篇介绍回调过滤器CallbackFilter。
一、作用
在CGLib回调时可以设置对不同方法执行不同的回调逻辑,或者根本不执行回调。
在JDK动态代理中并没有类似的功能,对InvocationHandler接口方法的调用对代理类内的所以方法都有效。
二、示例:
首先定义一个被代理类,包含3个方法:
- public class ConcreteClassNoInterface {
- public String getConcreteMethodA(String str){
- System.out.println("ConcreteMethod A ... "+str);
- return str;
- }
- public int getConcreteMethodB(int n){
- System.out.println("ConcreteMethod B ... "+n);
- return n+10;
- }
- public int getConcreteMethodFixedValue(int n){
- System.out.println("getConcreteMethodFixedValue..."+n);
- return n+10;
- }
- }
public class ConcreteClassNoInterface { public String getConcreteMethodA(String str){ System.out.println("ConcreteMethod A ... "+str); return str; } public int getConcreteMethodB(int n){ System.out.println("ConcreteMethod B ... "+n); return n+10; } public int getConcreteMethodFixedValue(int n){ System.out.println("getConcreteMethodFixedValue..."+n); return n+10; } }
其次,定义实现过滤器CallbackFilter接口的类:
- public class ConcreteClassCallbackFilter implements CallbackFilter{
- public int accept(Method method) {
- if("getConcreteMethodB".equals(method.getName())){
- return 0;//Callback callbacks[0]
- }else if("getConcreteMethodA".equals(method.getName())){
- return 1;//Callback callbacks[1]
- }else if("getConcreteMethodFixedValue".equals(method.getName())){
- return 2;//Callback callbacks[2]
- }
- return 1;
- }
- }
public class ConcreteClassCallbackFilter implements CallbackFilter{ public int accept(Method method) { if("getConcreteMethodB".equals(method.getName())){ return 0;//Callback callbacks[0] }else if("getConcreteMethodA".equals(method.getName())){ return 1;//Callback callbacks[1] }else if("getConcreteMethodFixedValue".equals(method.getName())){ return 2;//Callback callbacks[2] } return 1; } }
其中return值为被代理类的各个方法在回调数组Callback[]中的位置索引(见下文)。
生成代理类:
- Enhancer enhancer=new Enhancer();
- enhancer.setSuperclass(ConcreteClassNoInterface.class);
- CallbackFilter filter=new ConcreteClassCallbackFilter();
- enhancer.setCallbackFilter(filter);
- Callback interceptor=new ConcreteClassInterceptor();//(1)
- Callback noOp=NoOp.INSTANCE;//(2)
- Callback fixedValue=new ConcreteClassFixedValue();//(3)
- Callback[] callbacks=new Callback[]{interceptor,noOp,fixedValue};
- enhancer.setCallbacks(callbacks);
- ConcreteClassNoInterface proxyObject=(ConcreteClassNoInterface)enhancer.create();
- //...见下文
Enhancer enhancer=new Enhancer(); enhancer.setSuperclass(ConcreteClassNoInterface.class); CallbackFilter filter=new ConcreteClassCallbackFilter(); enhancer.setCallbackFilter(filter); Callback interceptor=new ConcreteClassInterceptor();//(1) Callback noOp=NoOp.INSTANCE;//(2) Callback fixedValue=new ConcreteClassFixedValue();//(3) Callback[] callbacks=new Callback[]{interceptor,noOp,fixedValue}; enhancer.setCallbacks(callbacks); ConcreteClassNoInterface proxyObject=(ConcreteClassNoInterface)enhancer.create(); //...见下文
生成代理类前,设置了CallbackFilter,上文中ConcreteClassCallbackFilter实现类的返回值对应Callback[]数组中的位置索引。此处包含了CGLib中的3种回调方式:
(1)MethodInterceptor:方法拦截器,上一篇文章中已经详细介绍过,此处不再赘述。
(2)NoOp.INSTANCE:这个NoOp表示no operator,即什么操作也不做,代理类直接调用被代理的方法不进行拦截。
(3)FixedValue:表示锁定方法返回值,无论被代理类的方法返回什么值,回调方法都返回固定值。
其中,ConcreteClassFixedValue类实现如下:
- public class ConcreteClassFixedValue implements FixedValue{
- public Object loadObject() throws Exception {
- System.out.println("ConcreteClassFixedValue loadObject ...");
- Object object=999;
- return object;
- }
- }
public class ConcreteClassFixedValue implements FixedValue{ public Object loadObject() throws Exception { System.out.println("ConcreteClassFixedValue loadObject ..."); Object object=999; return object; } }
该类实现FixedValue接口,同时锁定回调值为999(整型,CallbackFilter中定义的使用FixedValue型回调的方法为getConcreteMethodFixedValue,该方法返回值为整型)。
下面进行验证:
- //接上文...
- System.out.println("*** NoOp Callback ***");
- proxyObject.getConcreteMethodA("abcde");
- System.out.println("*** MethodInterceptor Callback ***");
- proxyObject.getConcreteMethodB(1);
- System.out.println("*** FixedValue Callback ***");
- int fixed1=proxyObject.getConcreteMethodFixedValue(128);
- System.out.println("fixedValue1:"+fixed1);
- int fixed2=proxyObject.getConcreteMethodFixedValue(256);
- System.out.println("fixedValue2:"+fixed2);
//接上文... System.out.println("*** NoOp Callback ***"); proxyObject.getConcreteMethodA("abcde"); System.out.println("*** MethodInterceptor Callback ***"); proxyObject.getConcreteMethodB(1); System.out.println("*** FixedValue Callback ***"); int fixed1=proxyObject.getConcreteMethodFixedValue(128); System.out.println("fixedValue1:"+fixed1); int fixed2=proxyObject.getConcreteMethodFixedValue(256); System.out.println("fixedValue2:"+fixed2);
控制台输出:
- *** NoOp Callback ***
- ConcreteMethod A ... abcde
- *** MethodInterceptor Callback ***
- Before :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int)
- ConcreteMethod B ... 1
- After :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int)
- ConcreteClassInterceptor end...
- *** FixedValue Callback ***
- ConcreteClassFixedValue loadObject ...
- fixedValue1:999
- ConcreteClassFixedValue loadObject ...
- fixedValue2:999
*** NoOp Callback *** ConcreteMethod A ... abcde *** MethodInterceptor Callback *** Before :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int) ConcreteMethod B ... 1 After :public int generic.cglib.proxy.ConcreteClassNoInterface.getConcreteMethodB(int) ConcreteClassInterceptor end... *** FixedValue Callback *** ConcreteClassFixedValue loadObject ... fixedValue1:999 ConcreteClassFixedValue loadObject ... fixedValue2:999
getConcreteMethodA对应CallbackFilter中定义的索引1,在Callback[]数组中使用的过滤为NoOp,因此直接执行了被代理方法。
getConcreteMethodB对应CallbackFilter中定义的索引0,在Callback[]数组中使用的过滤为MethodInterceptor,因此执行了方法拦截器进行拦截。
getConcreteMethodFixedValue对应CallbackFilter中定义的索引2,在Callback[]数组中使用的过滤为FixedValue,因此2次赋值128和256的调用其结果均被锁定为返回999。
相关推荐
Maven坐标:cglib:cglib-nodep:3.1; 标签:cglib、nodep、中文文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变...
3. `net.sf.cglib.proxy.Callback`: 这是CGlib回调接口的基类,包括MethodInterceptor和其他多种类型的回调。你可以根据需求选择合适的回调实现。 4. `net.sf.cglib.core.DefaultNamingPolicy` 和 `...
Cglib就是一种实现动态代理的方式,不同于JDK自带的Proxy,Cglib不需要目标对象实现任何接口,因此可以用于不能实现接口的对象。通过Enhancer类,我们可以指定需要代理的目标类,并提供回调方法实现动态代理逻辑。 ...
Maven坐标:cglib:cglib:3.1; 标签:cglib、中文文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持不变,注释和说明...
- **net.sf.cglib.proxy.Callback:** CGLIB 中的关键接口,所有回调接口都需要继承此接口。 - **net.sf.cglib.proxy.MethodInterceptor:** 最常用的回调类型,用于实现方法的拦截。它定义了一个方法: - `public ...
2. **接口回调**:对于接口,CGLib可以使用`Enhancer`类配合`Callback`接口来创建代理对象。这种方式更加灵活,但要求目标类必须实现至少一个接口。 关键组件和接口包括: - **Enhancer**:这是CGLib的主要入口点...
2. 设置回调函数,即MethodInterceptor,这是CGLib实现动态代理的关键,它定义了当代理对象调用方法时执行的逻辑。 3. 调用Enhancer的create()方法,生成代理对象。 在Spring AOP中,CGLib通过Enhancer和...
在实际应用中,理解CGLib的工作原理和如何正确使用回调是非常重要的。这不仅有助于提升代码的灵活性和可维护性,还能有效地利用AOP来解决复杂的问题,例如日志记录、性能分析、权限控制等。 由于CGLib是基于字节码...
Maven坐标:cglib:cglib-nodep:3.2.4; 标签:cglib、nodep、中文文档、jar包、java; 使用方法:解压翻译后的API文档,用浏览器打开“index.html”文件,即可纵览文档内容。 人性化翻译,文档中的代码和结构保持...
3. **MethodInterceptor**:这是CGlib中最常用的回调接口,通过实现intercept()方法,可以在目标方法执行前后添加自定义逻辑,比如AOP中的切面逻辑。 4. **ASM库**:CGlib底层使用ASM库来读取、修改和生成字节码,...
Cglib是一个强大的Java代码生成库,主要用于动态创建子类,是Java编程中不可或缺的工具之一,尤其在AOP(面向切面编程)和ORM(对象关系映射)框架中广泛应用。Cglib是由Eclipse的AspectJ项目开发团队维护的,它能够...
- 设置回调机制:通过setCallback()或setCallbacks()方法设置回调接口,通常是MethodInterceptor。 - 创建代理对象:通过create()方法创建代理对象,此时的代理对象已经具备了目标类的所有方法,并且可以拦截这些...
2. **Callback**:这是CGlib的回调接口,包括MethodInterceptor、FixedValue等。它们定义了在目标方法执行前后需要进行的操作,如记录日志、性能监控、事务管理等。 3. **MethodInterceptor**:这是最常用的回调...
2. 设置回调函数(Callback),这通常是MethodInterceptor接口的实现,用于处理代理方法的调用。 3. 使用Enhancer的create()方法生成代理对象,此时Cglib会自动完成子类的生成和实例化。 以下是一个简单的Cglib代理...
2. **Callback**:回调接口,它定义了CGLIB如何处理代理对象的方法调用。常见的Callback接口有MethodInterceptor、FixedValue等。其中,MethodInterceptor是最常用的,它允许你在方法调用前后执行自定义逻辑。 3. *...
二、CGLIB原理 CGLIB原理:动态生成一个要代理类的子类,子类重写要代理的类的所有不是final的方法。在子类中采用方法拦截的技术拦截所有父类方法的调用,顺势织入横切逻辑。它比使用java反射的JDK动态代理要快。 ...
当你看到"COULD NOT FIND cglib:cglib-2.2.2"这样的错误信息时,这意味着你的Maven或Gradle构建系统在尝试构建项目时无法找到这个特定版本的cglib依赖。这通常是由于以下原因造成的: 1. **本地Maven仓库中缺失该...
本篇文章将深入探讨这两种代理机制,尤其是Cglib代理。 首先,Java的动态代理基于Java的反射API实现,主要用于接口代理。当目标对象实现了至少一个接口时,我们可以使用`java.lang.reflect.Proxy`类来创建代理对象...