我在“aspectj”模式下使用 Spring 的声明性事务(@Transactional 注释)。在大多数情况下,它的工作原理与应有的完全一样,但对于其中一种情况却没有。我们可以调用它Lang
(因为它实际上是这样调用的)。
我已经能够将问题定位到加载时间编织器。通过在 aop.xml 中打开调试和详细日志记录,它列出了所有正在编织的类。Lang
日志中确实根本没有提到有问题的类。
然后我在 的顶部放了一个断点Lang
,导致 Eclipse 在Lang
加载类时挂起线程。这个断点在 LTW 编织其他类时被命中!所以我猜它要么尝试编织Lang
并且失败并且不输出它,或者其他一些类有一个引用,它Lang
在它实际有机会编织它之前强制它加载。
但是我不确定如何继续调试它,因为我无法以较小的规模重现它。关于如何继续的任何建议?
更新:也欢迎其他线索。例如,LTW 实际上是如何工作的?似乎发生了很多魔术。是否有任何选项可以从 LTW 获得更多调试输出?我目前有:
<weaver options="-XnoInline -Xreweavable -verbose -debug -showWeaveInfo">
我忘了汤姆之前提到过:spring-agent被用来允许 LTW,即InstrumentationLoadTimeWeaver
.
根据 Andy Clement 的建议,我决定检查 AspectJ 变压器是否通过了课程。我在 中放了一个断点,尽管它是由与其他类相同的类加载器(Jetty 的 WebAppClassLoader 的一个实例)加载的,但该类ClassPreProcessorAgent.transform(..)
似乎甚至从未到达该方法。Lang
然后我继续在InstrumentationLoadTimeWeaver$FilteringClassFileTransformer.transform(..)
. 甚至没有被击中Lang
。而且我相信应该为所有加载的类调用该方法,无论它们使用什么类加载器。这开始看起来像:
- 我的调试有问题。可能
Lang
在 Eclipse 报告它的时候没有加载 - 爪哇错误?有点牵强,但我想它确实会发生。
下一条线索:我打开了电源-verbose:class
,它似乎Lang
被过早地加载了——可能是在将变压器添加到 Instrumentation 之前。奇怪的是,我的 Eclipse 断点没有捕捉到这个加载。
这意味着 Spring 是新的嫌疑人。似乎有一些处理ConfigurationClassPostProcessor
加载类来检查它们。这可能与我的问题有关。
这些行ConfigurationClassBeanDefinitionReader
导致Lang
类被读取:
else if (metadata.isAnnotated(Component.class.getName()) ||
metadata.hasAnnotatedMethods(Bean.class.getName())) {
beanDef.setAttribute(CONFIGURATION_CLASS_ATTRIBUTE, CONFIGURATION_CLASS_LITE);
return true;
}
特别是对类的metadata.hasAnnotatedMethods()
调用getDeclaredMethods()
,它会加载该类中所有方法的所有参数类。我猜这可能不是问题的结束,因为我认为这些类应该被卸载。JVM 会因为不可知的原因缓存类实例吗?