Java CGLIB动态代理示例

1. CGLIB动态代理简介

JDK动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。
而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。

2. JDK动态代理与CGLIB区别

JDK动态代理和CGLIB字节码生成的区别?

  1. JDK动态代理只能对实现了接口的类生成代理,而不能针对类
  2. CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法。因为是继承,所以该类或方法最好不要声明成final
3.代码示例

注:需要引入相关jar包,下载cglib.jarasm.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

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值