这篇文章主要讲CGLib的一个应用方面——动态代理
CGLib 是什么
之前我们有讲到ASM,但它相当于是 jvm的 “汇编语言” 还是太底层,写起来很麻烦。于是有了CGLib
CGLib 是一个 可以动态修改 .class字节码的工具。它基于 ASM,在ASM的基础上做了封装,屏蔽了底层细节,并向上提高一些高级的API抽象。
CGLib封装程度高,简单、容易上手,因此名气反而必 ASM 还大。(充分告诉我们,简单易使用在任何领域都是多么的重要)
动态代理
cglib 的应用范围很广, 动态代理只是其中的一个应用,不要认为 cglib就只能做动态代理。
使用条件
目标类没有被 final 关键字修饰
原因
既然是代理,那肯定要涉及
- 目标类
- 代理类
而且代理类必须要和目标类是同一个类型!
什么意思
比如有下面这样一个方法,它的参数为 HelloWorld 类型
public void hello(HelloWorld helloWorld) {
return helloWorld.sayHello();
}
假设你要代理 HelloWorld 类中的 sayHello() 方法,你的代理类叫 HelloWorldProxy。如果HelloWorldProxy 和 HelloWorld 没有任何关系的话,那么它无法作为参数传递给hello()方法。
所以说 代理类必须要和目标类是同一个类型!
这里的“同一个类型”指的是
- 要么,代理类和目标类都实现了同一个接口,他们都属于同一种类型,即都是该接口的子类。
- 要么,代理类继承自目标类,这也是属于同一种类型,前提是目标类没有被 final 关键字修饰
从上面可以看到,两种方式都有限制。
第一种方式,依赖于目标类的接口,如果目标了没有接口,就无法实现。第二种方式,依赖于目标类没有被 final 关键字修饰
java动态代理技术就是基于第一种方式实现的。而cglib选择第二种方式,毕竟一个类没有实现接口的情况很多,但被 final 关键字修饰的情况很少。
原理
cglib可以在运行时,动态生成一个代理类继承我们的目标类,并重写了目标方法。
如下
动态生成的代理类,在代理方法中 调用了父类(目标类)的目标方法,并在调用前后做了一些处理。
CGLib的动态代理 与java动态代理的区别
- Java动态代理生成的代理类和目标类是 “兄弟”关系
- cglib动态代理生成的代理类和目标类是 “父子”关系
CGLib的动态代理 与java动态代理的局限性
- Java动态代理依赖于目标类有实现接口
- cglib动态代依赖于目标类没有被final关键字修饰
尾巴
CGLIb为了提高性能,还用了一种叫做FastClass的方式来直接调用一个对象的方法,而不是通过反射。 至于什么是 FastClass的方式,之后在写篇文章