79、Spring5.x源码之旅八十事务CGLIB动态代理的秘密二

  • CGLIB代理方法执行基本流程
  • 拦截器的intercept调用的秘密
    • MethodProxy的invokeSuper调用父类方法
    • MethodProxy的init创建FastClassInfo
      • FastClassInfo
      • 动态生成FastClass增强类

CGLIB代理方法执行基本流程

*

拦截器的intercept调用的秘密

一般我们在这里面做一些增强,然后调用原来的对象的方法,也就是methodProxy.invokeSuper,但是前面说了,传入参数o是代理对象,他是怎么调用原来的对象呢。:
*

MethodProxy的invokeSuper调用父类方法

其实看这个名字就知道是调用原方法,调用父类方法嘛,前面有继承嘛,对吧。但是事实是怎么样呢,我们来看看吧:

	public Object invokeSuper(Object obj, Object[] args) throws Throwable {
   
     
		try {
   
     
			init();//初始化fastClassInfo
			FastClassInfo fci = fastClassInfo;
			return fci.f2.invoke(fci.i2, obj, args);
		}
		catch (InvocationTargetException e) {
   
     
			throw e.getTargetException();
		}
	}

MethodProxy的init创建FastClassInfo

FastClassInfo

这个类其实就是存放了两个FastClass对象,和两个索引,他们是什么,其实就是两个FastClass增强对象,和锁对应方法的索引。索引什么鬼,简单的说吧,JDK反射调用方法有很多检查,性能肯定有损耗,所以CGLIB就想能不能我再创建两个类,一个是有原方法的类,一个是代理方法的类,里面有很多方法对应的索引,如果我能获取一个方法的索引,根据索引可以调用对应类的方法,这样避免用反射调用,具体做法我后面会分析,现在知道个大概就好。
*

动态生成FastClass增强类

这里就是根据创建的信息CreateInfo,来创建被代理类和代理类的FastClass增强对象,而且是动态创建的,还用了双重检测,只会创建一次。

private void init() {
   
     
		if (fastClassInfo == null) {
   
     
			synchronized (initLock) {
   
     
				if (fastClassInfo == null) {
   
     
					CreateInfo ci = createInfo;

					FastClassInfo fci = new FastClassInfo();
					fci.f1 = helper(ci, ci.c1);
					fci.f2 = helper(ci, ci.c2);
					fci.i1 = fci.f1.getIndex(sig1);
					fci.i2 = fci.f2.getIndex(sig2);
					fastClassInfo = fci;
					createInfo = null;
				}
			}
		}
	}

此时我们看到两个增强的类,和两个索引,f1,i1对应的是被代理类的FastClass增强对象,f2,i2对应的是代理类的FastClass增强对象:
*
而且生成了字节码文件,就是EnhancerByCGLIB再增强FastClassByCGLIB,另外一个CglibObj$$FastClassByCGLIB$$2bb5f2d7是在前面创建代理的时候就创建了,只是前面没讲,现在讲:
*
这些FastClass增强类到底增强什么呢,其实是方法的调用,跟索引有关,他会根据方法签名去FastClass增强类获取相关的索引,因为所有的方法都在FastClass增强类里:
*
看到了吧,CglibObj$$FastClassByCGLIB$$2bb5f2d7里面f1方法的返回索引是0
*
看到了吧,CglibObj$$EnhancerByCGLIB$$ef630afc$$FastClassByCGLIB$$9f694f5b里面f1的方法的返回索引是13
*
这些方法前面字符串哪里来的啊,其实前面有,我就稍微提下:
*
namedesc拼起来的,这些都是生成字节码的时候根据方法生成的:
*
*
FastClass增强对象和方法索引都获取到了:
*

获取了之后怎么用呢,下篇讲啦。

好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。