我想知道当你用@Transactional注释一个方法时实际发生了什么? 当然,我知道Spring会将该方法包装在一个事务中。

但是,我有以下疑问:

我听说Spring创建了一个代理类?有人能更深入地解释一下吗?代理类中实际驻留的是什么?实际的类发生了什么?以及如何查看Spring创建的代理类 我还在Spring文档中读到:

注意:由于这种机制是基于代理的,只有通过代理进入的“外部”方法调用才会被拦截。这意味着“自调用”,即目标对象中的方法调用目标对象中的其他方法,即使被调用的方法标记为@Transactional!

来源:http://static.springsource.org/spring/docs/2.0.x/reference/transaction.html

为什么只有外部方法调用将在事务下,而不是自调用方法?


当前回答

可能有点晚了,但我发现了一些东西,很好地解释了你对代理的担忧(只有通过代理进入的“外部”方法调用才会被拦截)。

例如,你有一个这样的类

@Component("mySubordinate")
public class CoreBusinessSubordinate {

    public void doSomethingBig() {
        System.out.println("I did something small");
    }

    public void doSomethingSmall(int x){
        System.out.println("I also do something small but with an int");    
  }
}

你有一个方面,看起来像这样:

@Component
@Aspect
public class CrossCuttingConcern {

    @Before("execution(* com.intertech.CoreBusinessSubordinate.*(..))")
    public void doCrossCutStuff(){
        System.out.println("Doing the cross cutting concern now");
    }
}

当你像这样执行它时:

 @Service
public class CoreBusinessKickOff {

    @Autowired
    CoreBusinessSubordinate subordinate;

    // getter/setters

    public void kickOff() {
       System.out.println("I do something big");
       subordinate.doSomethingBig();
       subordinate.doSomethingSmall(4);
   }

}

上面给定的代码调用kickOff的结果。

I do something big
Doing the cross cutting concern now
I did something small
Doing the cross cutting concern now
I also do something small but with an int

但是当你把代码改成

@Component("mySubordinate")
public class CoreBusinessSubordinate {

    public void doSomethingBig() {
        System.out.println("I did something small");
        doSomethingSmall(4);
    }

    public void doSomethingSmall(int x){
       System.out.println("I also do something small but with an int");    
   }
}


public void kickOff() {
  System.out.println("I do something big");
   subordinate.doSomethingBig();
   //subordinate.doSomethingSmall(4);
}

你看,这个方法在内部调用另一个方法,所以它不会被拦截,输出看起来像这样:

I do something big
Doing the cross cutting concern now
I did something small
I also do something small but with an int

你可以通过这样做来绕过这个问题

public void doSomethingBig() {
    System.out.println("I did something small");
    //doSomethingSmall(4);
    ((CoreBusinessSubordinate) AopContext.currentProxy()).doSomethingSmall(4);
}

代码片段取自: https://www.intertech.com/Blog/secrets-of-the-spring-aop-proxy/

其他回答

所有现有的答案都是正确的,但我觉得不能公正地对待这个复杂的话题。

要获得全面、实用的解释,您可能想要看看Spring @Transactional深度指南,它尽量用大约4000个简单的单词介绍事务管理,并提供了大量的代码示例。

作为一个视觉型的人,我喜欢用代理模式的序列图来衡量。如果你不知道如何阅读箭头,我是这样阅读第一个:客户端执行Proxy.method()。

客户端从他的角度调用目标上的一个方法,并被代理静默地拦截 如果定义了before方面,代理将执行它 然后,执行实际的方法(目标) 后返回和后投掷是可选的 方法返回和/或方法抛出 异常 之后,代理执行After方面(如果已定义) 最后,代理返回到调用客户端

(我被允许发布这张照片,条件是我必须提及照片的来源。作者:诺埃尔·维斯,网址:https://www.noelvaes.eu)

当Spring加载bean定义并配置为查找@Transactional注释时,它将围绕实际的bean创建这些代理对象。这些代理对象是在运行时自动生成的类的实例。当调用方法时,这些代理对象的默认行为只是在“目标”bean(即您的bean)上调用相同的方法。

However, the proxies can also be supplied with interceptors, and when present these interceptors will be invoked by the proxy before it invokes your target bean's method. For target beans annotated with @Transactional, Spring will create a TransactionInterceptor, and pass it to the generated proxy object. So when you call the method from client code, you're calling the method on the proxy object, which first invokes the TransactionInterceptor (which begins a transaction), which in turn invokes the method on your target bean. When the invocation finishes, the TransactionInterceptor commits/rolls back the transaction. It's transparent to the client code.

至于“外部方法”,如果您的bean调用它自己的方法之一,那么它将不会通过代理进行调用。请记住,Spring将bean包装在代理中,您的bean不知道它。只有来自bean“外部”的调用才会通过代理。

这有用吗?

可能有点晚了,但我发现了一些东西,很好地解释了你对代理的担忧(只有通过代理进入的“外部”方法调用才会被拦截)。

例如,你有一个这样的类

@Component("mySubordinate")
public class CoreBusinessSubordinate {

    public void doSomethingBig() {
        System.out.println("I did something small");
    }

    public void doSomethingSmall(int x){
        System.out.println("I also do something small but with an int");    
  }
}

你有一个方面,看起来像这样:

@Component
@Aspect
public class CrossCuttingConcern {

    @Before("execution(* com.intertech.CoreBusinessSubordinate.*(..))")
    public void doCrossCutStuff(){
        System.out.println("Doing the cross cutting concern now");
    }
}

当你像这样执行它时:

 @Service
public class CoreBusinessKickOff {

    @Autowired
    CoreBusinessSubordinate subordinate;

    // getter/setters

    public void kickOff() {
       System.out.println("I do something big");
       subordinate.doSomethingBig();
       subordinate.doSomethingSmall(4);
   }

}

上面给定的代码调用kickOff的结果。

I do something big
Doing the cross cutting concern now
I did something small
Doing the cross cutting concern now
I also do something small but with an int

但是当你把代码改成

@Component("mySubordinate")
public class CoreBusinessSubordinate {

    public void doSomethingBig() {
        System.out.println("I did something small");
        doSomethingSmall(4);
    }

    public void doSomethingSmall(int x){
       System.out.println("I also do something small but with an int");    
   }
}


public void kickOff() {
  System.out.println("I do something big");
   subordinate.doSomethingBig();
   //subordinate.doSomethingSmall(4);
}

你看,这个方法在内部调用另一个方法,所以它不会被拦截,输出看起来像这样:

I do something big
Doing the cross cutting concern now
I did something small
I also do something small but with an int

你可以通过这样做来绕过这个问题

public void doSomethingBig() {
    System.out.println("I did something small");
    //doSomethingSmall(4);
    ((CoreBusinessSubordinate) AopContext.currentProxy()).doSomethingSmall(4);
}

代码片段取自: https://www.intertech.com/Blog/secrets-of-the-spring-aop-proxy/

最简单的答案是:

无论在哪个方法上声明@Transactional,当方法完成时,事务的边界开始并结束。

如果您正在使用JPA调用,那么所有提交都在此事务边界内。

假设你要保存entity1、entity2和entity3。现在,当保存entity3时发生异常,然后当entity1和entity2进入同一事务时,entity1和entity2将与entity3一起回滚。

事务:

entity1.save entity2.save entity3.save

任何异常都将导致与DB的所有JPA事务的回滚。Spring在内部使用JPA事务。