cglib FastClass机制
前言
关于动态代理的一些知识,以及cglib与jdk动态代理的区别,在这一篇已经介绍过,不熟悉的可以先看下。
本篇我们来学习一下cglib的FastClass机制,这是cglib与jdk动态代理的一个主要区别,也是一个面试考点。
我们知道jdk动态代理是使用InvocationHandler接口,在invoke方法内,可以使用Method方法对象进行反射调用,反射的一个最大问题是性能较低,cglib就是通过使用FastClass来优化反射调用,提升性能,接下来我们就看下它是如何实现的。
示例
我们先写一个hello world,让代码跑起来。如下:
public class HelloWorld {
public void print() {
System.out.println("hello world");
}
}
public class HelloWorldInterceptor implements MethodInterceptor {
public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
System.out.println("before hello world");
methodProxy.invokeSuper(o, objects);
System.out.println("after hello world");
return null;
}
}
非常简单,就是使用MethodInterceptor在HelloWorld类print方法前后打印一句话,模拟对一个方法前后织入自定义逻辑。
接着使用cglib Enhancer类,创建动态代理对象,设置MethodInterceptor,调用方法。
为了方便观察源码,我们将cglib生成的动态代理类保存下来。
//将生成的动态代理类保存下来
System.setProperty(DebuggingClassWriter.DEBUG_LOCATION_PROPERTY, "D:\\");
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(HelloWorld.class);
enhancer.setCallback(new HelloWorldInterceptor());
HelloWorld target = (HelloWorld) enhancer.create();
target.print();
输出
before hello world
hello world
after hello world
FastClass机制
我们知道cglib是通过继承实现的,动态代理类会继承被代理类,并重写它的方法,所以它不需要像jdk动态代理一样要求被代理对象有实现接口,因此比较灵活。
既然是通过继承实现的,那应该生成一个类就可以了,但是通过上面的路径观察,可以看到生成了3个文件,其中两个带有FastClass关键字。
这三个类分别是:动态代理类,动态代理类的FastClass,被代理对象的FastClass,从名称上也可以看出它们的关系。

其中动态代理类继承了被代理类,并重写了父类的所有方法,包括父类的父类的方法,包括Object类的equals方法和toString方法等。
public class HelloWorld$$EnhancerByCGLIB$$49f9f9c8 extends HelloWorld implements Factory {
}
这里我们只关注print方法,如下:

第一个直接调用父类方法,也就是被代理对象的方法;第二个会先判断有没有拦截器,如果没有也是直接调用父类方法,否则调用MethodInterceptor的intercept方法,对于我们这里就是HelloWorldInterceptor。
看下intercept的几个参数分别是什么,这几个参数的初始化在动态代理类的静态代码块中都可以找到。
第1个表示动态代理对象。
第2个是被代理对象方法的Method,就是HelloWorld.print。
第3个表示方法参数。
第4个是MethodProxy对象,通过名字我们可以知道它是方法的代理,每一个方法都会有一个对应的MethodProxy,它包含被代理对象、代理对象、以及对应的方法元信息。
这里我们重点关注MethodProxy,它的初始化如下:
CGLIB$print$0$Proxy = MethodProxy.create(var1, var0, "()V", "print", "CGLIB$print$0");
第1个参数表示被代理对象的Class。
第2个参数表示动态代理对象的Class。
第3个参数是方法的返回值。
第4个参数表示被代理对象的方法名称。
第5个参数表示对应动态代理对象的方法名称。
MethodProxy对象创建好后,我们上面就是通过它进行调用的
methodProxy.invokeSuper(o, objects);
invokeSuper主要源码如下:
public Object invokeSuper(Object obj, Object[] args) throws Throwable {
init();
FastClassInfo fci = fastClassInfo;
return fci.f2.invoke(fci.i2, obj, args);
}
private void init()
{
if (fastClassInfo == null)
{
synchronized (initLock)
{
if (fastClassInfo == null)
{
CreateInfo ci = createInfo;
FastClassInfo fci = new FastClassInfo();
fci.f1 = helper(ci, ci.c1); //被代理对象的FastClass
fci.f2 = helper(ci, ci.c2); //动态代理对象的FastClass
fci.i1 = fci.f1.getIndex(sig1); //被代理对象方法的索引下标
fci.i2 = fci.f2.getIndex(sig2); //动态代理对象方法的索引下标,这里是:CGLIB$print$0
fastClassInfo = fci;
createInfo = null;
}
}
}
}
init方法使用加锁+双检查的方式,只会初始化一次fastClassInfo变量,它用volatile关键字进行修饰,这里涉及到java字节码重排问题,具体可以参考我们之前的分析:happend before原则
接着回到invokeSuper方法,fci.f2.invoke(fci.i2, obj, args); 实际就是调用动态代理对象的FastClass的invoke方法,并把要调用方法的索引下标i2传过去。
至于方法的索引下标是怎么找到的,可以看动态代理对象的FastClass的getIndex方法,其实就是通过方法的名称、参数个数、参数类型,完全匹配,点到源码文件可以看到有大量的switch分支判断。
这里我们可以看到print方法的索引下标就是18。
public int getIndex(String var1, Class[] var2) {
switch (var1.hashCode()) {
case -1295482945:
if (var1.equals("equals")) {
switch (var2.length) {
case 1:
if (var2[0].getName().equals("java.lang.Object")) {
return 0;
}
}
}
break;
case 770871766:
if (var1.equals("CGLIB$print$0")) {
switch (var2.length) {
case 0:
return 18;
}
}
break;
}
}
public Object invoke(int var1, Object var2, Object[] var3) throws InvocationTargetException {
HelloWorld..EnhancerByCGLIB..49f9f9c8 var10000 = (HelloWorld..EnhancerByCGLIB..49f9f9c8)var2;
int var10001 = var1;
//...
switch (var10001) {
//...
case 18:
var10000.CGLIB$print$0();
return null;
}
}
可以看到最终调用到动态代理类的CGLIB$print$0方法,也就是:
final void CGLIB$print$0() {
super.print();
}
最终调用的就是父类的方法。我们画张图总结一下,有兴趣的同学跟着图和代码逻辑应该可以快速理解。

总结
经过上面的分析,我们可以看到cglib在整个调用过程并没有用到反射,而是使用FastClass对每个方法进行索引,通过方法名称,参数长度,参数类型就可以找到具体的方法,因此性能较好。但也有缺点,首次调用需要生成3个类,会比较慢。在我们实际开发中,特别是一些框架开发,如果有类似的场景也可以借助FastClass对反射进行优化,如:
MyClass cs = new MyCase();
FastClass fastClass = FastClass.create(Case.class);
int index = fastClass.getIndex("test", new Class[]{Integer.class});
Object invoke = fastClass.invoke(index, cs, new Object[1]);
另外MethodProxy还有一个invoke方法,如果我们换一下调用这个方法会发生?留给大家自己尝试。
methodProxy.invokeSuper(o, objects);
//换成 methodProxy.invoke(o, objects);
cglib FastClass机制的更多相关文章
- 详解Java动态代理机制(二)----cglib实现动态代理
上篇文章的结尾我们介绍了普通的jdk实现动态代理的主要不足在于:它只能代理实现了接口的类,如果一个类没有继承于任何的接口,那么就不能代理该类,原因是我们动态生成的所有代理类都必须继承Proxy这个类, ...
- 基于 CGLIB 库的动态代理机制
之前的文章我们详细的介绍了 JDK 自身的 API 所提供的一种动态代理的实现,它的实现相对而言是简单的,但是却有一个非常致命性的缺陷,就是只能为接口中的方法完成代理,而委托类自己的方法或者父类中的方 ...
- cglib源码分析(四):cglib 动态代理原理分析
本文分下面三个部分来分析cglib动态代理的原理. cglib 动态代理示例 代理类分析 Fastclass 机制分析 一.cglib 动态代理示例 public class Target{ publ ...
- java 动态代理模式(jdk和cglib)
package proxy.dynamicproxy; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Met ...
- JDK动态代理和cglib代理详解
JDK动态代理 先做一下简单的描述,通过代理之后返回的对象已并非原类所new出来的对象,而是代理对象.JDK的动态代理是基于接口的,也就是说,被代理类必须实现一个或多个接口.主要原因是JDK的代理原理 ...
- Cglib动态代理实现方式
Cglib动态代理实现方式 我们先通过一个demo看一下Cglib是如何实现动态代理的. 首先定义个服务类,有两个方法并且其中一个方法用final来修饰. public class PersonSer ...
- Cglib动态代理实现原理
Cglib动态代理实现方式 我们先通过一个demo看一下Cglib是如何实现动态代理的. 首先定义个服务类,有两个方法并且其中一个方法用final来修饰. public class PersonSer ...
- cglib实现动态代理简单使用
Boss: package proxy.cglib; public class Boss{ public void findPerson() { System.out.println("我要 ...
- 动态代理(二)—— CGLIB代理原理
前篇文章动态代理(一)--JDK中的动态代理中详细介绍了JDK动态代理的Demo实现,api介绍,原理详解.这篇文章继续讨论Java中的动态代理,并提及了Java中动态代理的几种实现方式.这里继续介绍 ...
- jdk 动态代理和 cglib 动态代理
原理区别: java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理. 而cglib动态代理是利用asm开源包,对代理对象类的class文件加载 ...
随机推荐
- NetCat 工具的常用使用技巧
netcat 黑客们的瑞士军刀,虽然小巧但是其功能一点也不弱,并且该工具天生免杀,值得你去尝试. NCwindows反弹 1:正向连接 服务器执行:nc -l -p 8888 -e cmd.exe 本 ...
- intel更新13代桌面处理器产品线,更多核心更贵价格
intel 13代酷睿产品线更新带来了更多低于125w TDP的处理器.intel确认了10个65wTDP的SKU和6个35wTDP的SKU.13代酷睿包含新的Raptor Lake和旧的Alder ...
- 【C语言深度解剖】预定义章节经典面试题讲解(offsetof宏模拟实现)【超详细的解释和注释】
[C语言深度解剖]预定义章节经典面试题讲解(offsetof宏模拟实现)[超详细的解释和注释] 那么这里博主先安利一下一些干货满满的专栏啦! 作者: #西城s 这是我的主页:#西城s 在食用这篇博客之 ...
- Promise, async, await实现异步编程,代码详解
写在开头 一点题外话 其实最近在不断的更新Java的知识,从基础到进阶,以及计算机基础.网络.WEB.数据库.数据结构.Linux.分布式等等内容,预期写成一个既可以学习提升又可以面试找工作的< ...
- macOS 上 常用的操作
首先 mac上 若使用的是windows的键盘,那么需要把ctrl 键,设置成 cmd键,因为mac上大多数操作都是 基于cmd键. 1.将ctrl键,修改为cmd键,这样后 复制.粘贴.剪切.全选等 ...
- 聚石塔容器查看tomcat 日志的方法
通过以上命令可以看出日志的路径,从而得出直接执行的命令:tail -f acs/log/catalina.log
- 《AI驱动下的开发者新生态》-2024长沙.NET技术社区活动-诚邀大家报名
回顾 2019年初,在.NET中文社区及包括苏州.广州.深圳等地区社区等大力推动.在众多企业的大力支持下,长沙地区的开发者们发起成立了长沙.NET技术社区,并组织了<2019年长沙开发者技术大会 ...
- 扯淡的DevOps,我们开发根本不想做运维!
引言 最初考虑引用" DevOps 已死,平台工程才是未来"作为标题,但这样的表达可能太过于绝对.最终,决定用了"扯淡的"这个词来描述 DevOps,但这并不是 ...
- NC23482 小A的最短路
题目链接 题目 题目描述 小A这次来到一个景区去旅游,景区里面有N个景点,景点之间有N-1条路径.小A从当前的一个景点移动到下一个景点需要消耗一点的体力值.但是景区里面有两个景点比较特殊,它们之间是可 ...
- 修改文件权限后Git 文件目录被标记为修改
刚打开IDE,工作区的代码状态全部变成修改未提交的状态了?这是这么回事?这是因为Git忽略文件权限或者拥有者改变导致的git状态变化.默认Git会记录文件的权限信息,如果文件的权限信息被修改,在Git ...