转载

CGLib 动态代理

之前我们有讲到ASM,但它相当于是 jvm的 “汇编语言” 还是太底层,写起来很麻烦。于是有了CGLib

CGLib 是一个 可以动态修改 .class字节码的工具。它基于 ASM,在ASM的基础上做了封装,屏蔽了底层细节,并向上提高一些高级的API抽象。

CGLib封装程度高,简单、容易上手,因此名气反而必 ASM 还大。(充分告诉我们,简单易使用在任何领域都是多么的重要)

动态代理

cglib 的应用范围很广, 动态代理只是其中的一个应用,不要认为 cglib就只能做动态代理。

使用条件

既然是代理,那肯定要涉及

  • 目标类
  • 代理类

而且代理类必须要和目标类是同一个类型!

什么意思

比如有下面这样一个方法,它的参数为 HelloWorld 类型

public void hello(HelloWorld helloWorld) {
    return helloWorld.sayHello();
}
复制代码

假设你要代理 HelloWorld 类中的 sayHello() 方法,你的代理类叫 HelloWorldProxy。如果HelloWorldProxy 和 HelloWorld 没有任何关系的话,那么它无法作为参数传递给hello()方法。

所以说 代理类必须要和目标类是同一个类型!

这里的“同一个类型”指的是

  • 要么, 代理类目标类 都实现了同一个接口,他们都属于同一种类型,即都是该接口的子类。
  • 要么, 代理类 继承自 目标类 ,这也是属于同一种类型,前提是目标类没有被 final 关键字修饰

从上面可以看到,两种方式都有限制。

第一种方式,依赖于目标类的接口,如果目标了没有接口,就无法实现。第二种方式,依赖于目标类没有被 final 关键字修饰

java动态代理技术就是基于第一种方式实现的。而cglib选择第二种方式,比较一个类没有接口的情况很多,但被 final 关键字修饰的情况很少。

原理

cglib可以在运行时,动态生成一个代理类继承我们的目标类,并重写了目标方法。

如下

CGLib 动态代理

动态生成的代理类,在代理方法中 调用了父类(目标类)的目标方法,并在调用前后做了一些处理。

原文  https://juejin.im/post/5ccd2b9df265da03761e9b65
正文到此结束
Loading...