cglib使用
源码分析
入口为new Enhancer()
Enhancer实例化时会给静态属性KEY_FACTORY赋值,该对象主要作为后续对生成的class文件缓存时生成key,该属性较为重要,后续会用到
KEY_FACTORY的类型是EnhancerKey,是个接口,所以第一步是为该类型生成代理class类。
KeyFactory.create()
实例化出来的Generator是真正生成对象的类,这部分代码主要是对其进行配置
gen.create()
调用父类(AbstractClassGenerator)的create方法
super.create()
这个方法是重点方法。
因为是第一次进这个方法,所以缓存中没有值。这边代码主要就是生成缓存对象。然后调用data.get方法生成类。
该类中有个Map<ClassLoader, ClassLoaderData>类型的CACHE,其中key是类加载器。value是ClassLoaderData。
ClassLoaderData中有两个重要的函数式接口,一个是Function<AbstractClassGenerator, Object> GET_KEY,该接口作用是返回一个用于缓存生成类的key,另一个是Function<AbstractClassGenerator, Object> load,作用是生成代理类字节码并加载到内存。
LoadingCache实例化时,将GET_KEY赋值到keyMapper,load赋值为loader。这两个后续会用到,请读者留意。
data.get()
默认使用缓存,走第二个分支
generatedClasses.get()
首先获取缓存的key,这里的keyMapper是之前通过构造方法赋值的,是一个接口,返回一个key。然后通过key去获取缓存,如果没有则生成新的class文件。
this.createEntry()
这里有点饶人,其实就是调用了一下loader的方法生成class实例。
LoadingCache.this.loader.apply()->gen.generate()
核心生成class的方法是strategy.generate()
strategy.generate()
观察者模式,创建类
this.transform(cg).generateClass()
这边是具体如何生成字节码文件的地方,读者可以按需阅读。
以上步骤仅仅是生成了一个keyFactory接口的class文件,并且实例化赋值给了静态变量。接下来才是增强类的生成过程。
enhancer.create()
createHelper()
利用之前生成的KEY_FACTORY的对象生成key,调用父类的create方法
super.create()
再次回到AbstractClassGenerator.create()方法,后续步骤跟上文一样。
注意事项
源码动态生成了两个代理类。一个是KeyFactory,另一个才是我们自己的增强类。
增强类分析
sayHello()
可以看到我们自己的方法sayHello在子类中被重写了,增加了调用InvocationHandler的逻辑。
在调用之前先绑定了callbacks
CGLIB$BIND_CALLBACKS()
绑定逻辑是从CGLIB$THREAD_CALLBACKS中get,那么又是如何将拦截器设置到CGLIB$THREAD_CALLBACKS中的呢
CGLIB$SET_THREAD_CALLBACKS()
可以看到是实例化增强类的时候,设置了ThreadCallbacks。