我想知道当您使用 @Transactional
注释方法时实际发生了什么?当然,我知道 Spring 会将该方法包装在 Transaction 中。
但是,我有以下疑问:
- 听说 Spring 创建了一个代理类?有人可以更深入解释这一点。 该代理类中实际存在什么?实际类(class)会发生什么?以及如何查看 Spring 创建的代理类
- 我还在 Spring 文档中读到:
Note: Since this mechanism is based on proxies, only 'external' method calls coming in through the proxy will be intercepted. This means that 'self-invocation', i.e. a method within the target object calling some other method of the target object, won't lead to an actual transaction at runtime even if the invoked method is marked with @Transactional
!
来源:http://static.springsource.org/spring/docs/2.0.x/reference/transaction.html
为什么 Transaction 下只有外部方法调用而不是自调用方法?
这是一个很大的话题。 Spring 引用文档用多个章节介绍它。我建议阅读 Aspect-Oriented Programming 上的内容和 Transactions ,因为 Spring 的声明式事务支持在其基础上使用 AOP。
但在非常高的层次上,Spring 为在类本身或成员上声明 @Transactional
的类创建代理。代理在运行时大部分是不可见的。它为 Spring 提供了一种在方法调用之前、之后或周围将行为注入(inject)被代理的对象的方法。事务管理只是可以 Hook 的行为的一个例子。安全检查是另一个例子。你也可以提供你自己的,比如日志记录。因此,当您使用 @Transactional
注释方法时,Spring 会动态创建一个代理,该代理实现与您正在注释的类相同的接口(interface)。并且当客户端调用你的对象时,调用会被拦截并通过代理机制注入(inject)行为。
顺便说一下,EJB 中的事务的工作方式类似。
正如您所观察到的,代理机制仅在调用来自某个外部对象时才起作用。当您在对象中进行内部调用时,您实际上是通过绕过代理的 this
引用进行调用。但是,有一些方法可以解决这个问题。我在 this forum post 中解释了一种方法在其中,我使用 BeanFactoryPostProcessor
在运行时将代理实例注入(inject)“自引用”类。我将此引用保存到名为me
的成员变量。然后,如果我需要进行需要更改线程事务状态的内部调用,我会通过代理直接调用(例如 me.someMethod()
。)论坛帖子更详细地解释了.
请注意,BeanFactoryPostProcessor
代码现在会有所不同,因为它是在 Spring 1.x 时间范围内写回的。但希望它能给你一个想法。我有一个更新版本,我可能会提供。
我是一名优秀的程序员,十分优秀!