在类和方法上定义@Transactional有什么区别


74

情况1

@Transactional
public class UserServiceImpl implements UserService {

    ...................
    public void method1(){
        try{
            method2();
        }catch(Exception e){

        }
    }
    public void method2(){

    }
}

案例2

public class UserServiceImpl implements UserService {

    ...................
    public void method1(){
        try{
            method2();
        }catch(Exception e){

        }
    }
    @Transactional
    public void method2(){

    }
}

在情况1中,如果发生任何异常,则回滚有效,但在情况2中,回滚无效。如果遵循case1,是否有任何性能问题?

Answers:


84

在情况1中,@ Transactional应用于每个公共的单个方法。Spring会忽略私有和受保护的方法。

Spring将类级别的注释应用于我们未使用@Transactional注释的此类的所有公共方法。但是,如果我们将注释放在私有或受保护的方法上,Spring将忽略它而不会出现错误。

在情况2中,@ Transactional仅应用于method2(),而不应用于method1()

情况1:-调用method1()->事务开始。当method1()调用method2()时,不会启动新事务,因为已经有一个

情况2:-调用method1()->没有事务开始。当方法1()调用方法2()NO启动新的事务。这是因为当从同一类中调用方法时,@ Transactional不起作用。如果您要从另一个类调用method2(),它将起作用。

弹簧参考手册

在代理模式(默认设置)下,仅拦截通过代理传入的外部方法调用。这意味着自调用实际上是目标对象中调用目标对象另一种方法的方法,即使调用的方法标有@Transactional,也不会在运行时导致实际事务。另外,必须完全初始化代理以提供预期的行为,因此您不应在初始化代码(即@PostConstruct)中依赖此功能。


要从非带注释的方法调用带注释的方法的警告加倍:即使对于带注释的方法,它也不会启动事务。另请参阅莫的答案。
kaicarno

41

@Transactional类上的on适用于服务上的每个方法。这是捷径。通常,@Transactional(readOnly = true)如果您知道所有方法都将访问存储库层,则可以在服务类上进行设置。然后,可以使用@Transactionalon方法在模型中进行更改来覆盖行为。1)和2)之间的性能问题未知。


如果我注释类Transactional(value =“ notPrimaryTransactionManager”),然后使用Transactional(readOnly = true)注释成员方法,该方法将使用“ notPrimaryTransactionManager”还是使用spring提供的默认事务管理器?
Anjil Dhamala

22

假设您有以下课程:

@Transactional(readOnly = true)
public class DefaultFooService implements FooService {

  public Foo getFoo(String fooName) {
    // do something
  }

  // these settings have precedence for this method
  @Transactional(readOnly = false, propagation = Propagation.REQUIRES_NEW)
  public void updateFoo(Foo foo) {
    // do something
  }
}

@Transactional类级别的注释将应用于该类中的每个方法。

但是,当方法带有@Transactional(如updateFoo(Foo foo))注释时,它将优先于在类级别定义的事务设置。

更多信息:


5
但仅当外部调用updateFoo时。如果getFoo调用updateFoo(如询问的OP),则整个事务将为只读。
David Newcomb

18

这里报价

Spring团队的建议是,仅使用@Transactional注释对具体的类进行注释,而不是对接口进行注释。

由于此机制基于代理,因此将仅拦截通过代理传入的“外部”方法调用。这意味着即使调用的方法标记有@Transactional,“自调用”(即目标对象中的方法调用目标对象的其他方法)也不会在运行时导致实际事务!


8
这并没有指出带@Transactional注释的方法和带@Transactional注释的类之间的区别。
Konstantin Yovkov 2014年

1
@kocko>我以为OP将遵循该链接,不过我刚刚更新了答案,请看一看。
mok

1
@mok我遇到了“自我调用将不会创建交易”的问题,这个答案让我想起了一段时间后的学习。谢谢!
asgs '18

这并不能解释OP所要求的差异,而只是通过注释方法来突出显示一个难题@Transactional。链接的文档现已更新。为了后人的缘故,最新文档说:Spring团队建议您仅使用@Transactional注释对具体类(以及具体类的方法)进行注释,而不是对接口进行注释。
railomaya
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.