`

Spring @Transactional工作原理

 
阅读更多
原文地址:http://www.importnew.com/12300.html



本文将深入研究Spring的事务管理。主要介绍@Transactional在底层是如何工作的。之后的文章将介绍:

propagation(事务传播)和isolation(隔离性)等属性的使用
事务使用的陷阱有哪些以及如何避免
JPA和事务管理

很重要的一点是JPA本身并不提供任何类型的声明式事务管理。如果在依赖注入容器之外使用JPA,事务处理必须由开发人员编程实现。

1
2
3
4
5
6
7
8
9
10
11
12
UserTransaction utx = entityManager.getTransaction();

    try {
        utx.begin();

        businessLogic();

        utx.commit();
    } catch(Exception ex) {
        utx.rollback();
        throw ex;
    }
这种方式的事务管理使事务范围可以在代码中很清晰地表达出来,但它有以下缺点:

容易出现重复代码和错误
任何错误可能产生较大的影响
错误难以调试和复现
降低了代码库的可读性
如果该方法调用了其他的事务方法如何处理呢?
使用Spring @Transactional

使用Spring @Transactional,上面的代码就简化为:

1
2
3
4
@Transactional
    public void businessLogic() {
        ... use entity manager inside a transaction ...
    }
代码更加简洁,可读性更好,也是目前Spring中事务处理的推荐方式。

通过使用@Transactional,事务传播等很多重要方面可以自动处理。这种情况下如果businessLogic()调用了其他事务方法,该方法将根据选项确定如何加入正在运行事务。

这个强大机制的一个潜在缺点是它隐藏了底层的运行,当它不能正常工作时很难调试。

@Transactional含义

关于@Transactional,关键点之一是要考虑两个独立的概念,它们都有各自的范围和生命周期:

persistence context(持久化上下文)
database transaction(事务)
@Transactional本身定义了单个事务的范围。这个事务在persistence context的范围内。

JPA中的持久化上下文是EntityManager,内部实现使用了Hibernate Session(使用Hibernate作为持久化provider)。

持久化上下文仅仅是一个同步对象,它记录了有限集合的Java对象的状态,并且保证这些对象的变化最终持久化到数据库。

这是与单个事务非常不同的概念。一个Entity Manager可以跨越多个事务使用,而且的确是这样使用的。

EntityManager何时跨越多个事务?

最常见的情况是应用使用Open Session In View模式处理懒初始化异常时,之前的文章介绍过这种做法的优势和劣势。

这种情况下视图层运行的多个查询处于独立的事务中,而不是单事务的业务逻辑,但这些查询由相同的entity manager管理。

另一种情况是开发人员将持久化上下文标记为PersistenceContextType.EXTENDED,这表示它能够响应多个请求。

如何定义EntityManager和Transaction之间的关系?

这由应用开发者来选择,但是JPA Entity Manager最常用的方式是“Entity Manager per application transaction”(每个事务都有自己的实体管理器)模式。entity manager注入的常用方法是:

1
2
@PersistenceContext
    private EntityManager em;
这里默认为“Entity Manager per transaction”模式。这种模式下如果在@Transactional方法内部使用该Entity Manager,那么该方法将在单一事务中运行。

@PersistenceContext如何工作?

随之而来的问题就是@PersistenceContext如何仅在容器启动时注入entity manager,假定entity manager生命周期很短暂,而且每次请求需要多个entity manager。

答案是它不能:EntityManager是一个接口,注入到spring bean中的不是entity manager本身,而是在运行时代理具体entity manager的context aware proxy(上下文感知代理)。

通常用于代理的具体类为SharedEntityManagerInvocationHandler,借助调试器可以确认这一点。

那么@Transactional如何工作?

实现了EntityManager接口的持久化上下文代理并不是声明式事务管理的唯一部分,事实上包含三个组成部分:

EntityManager Proxy本身
事务的切面
事务管理器
看一下这三部分以及它们之间的相互作用。

事务的切面

事务的切面是一个“around(环绕)”切面,在注解的业务方法前后都可以被调用。实现切面的具体类是TransactionInterceptor。

事务的切面有两个主要职责:

在’before’时,切面提供一个调用点,来决定被调用业务方法应该在正在进行事务的范围内运行,还是开始一个新的独立事务。
在’after’时,切面需要确定事务被提交,回滚或者继续运行。
在’before’时,事务切面自身不包含任何决策逻辑,是否开始新事务的决策委派给事务管理器完成。

事务管理器

事务管理器需要解决下面两个问题:

新的Entity Manager是否应该被创建?
是否应该开始新的事务?
这些需要事务切面’before’逻辑被调用时决定。事务管理器的决策基于以下两点:

事务是否正在进行
事务方法的propagation属性(比如REQUIRES_NEW总要开始新事务)
如果事务管理器确定要创建新事务,那么将:

创建一个新的entity manager
entity manager绑定到当前线程
从数据库连接池中获取连接
将连接绑定到当前线程
使用ThreadLocal变量将entity manager和数据库连接都绑定到当前线程。

事务运行时他们存储在线程中,当它们不再被使用时,事务管理器决定是否将他们清除。

程序的任何部分如果需要当前的entity manager和数据库连接都可以从线程中获取。

EntityManager proxy

EntityManager proxy(前面已经介绍过)就是谜题的最后一部分。当业务方法调用entityManager.persist()时,这不是由entity manager直接调用的。

而是业务方法调用代理,代理从线程获取当前的entity manager,前面介绍过事务管理器将entity manager绑定到线程。

了解了@Transactional机制的各个部分,我们来看一下实现它的常用Spring配置。

整合三个部分

如何将三个部分组合起来使事务注解可以正确地发挥作用呢?首先定义entity manager工厂。

这样就可以通过持久化上下文注解注入Entity Manager proxy。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
@Configuration
    public class EntityManagerFactoriesConfiguration {
        @Autowired
        private DataSource dataSource;

        @Bean(name = "entityManagerFactory")
        public LocalContainerEntityManagerFactoryBean emf() {
            LocalContainerEntityManagerFactoryBean emf = ...
            emf.setDataSource(dataSource);
            emf.setPackagesToScan(
                new String[] {"your.package"});
            emf.setJpaVendorAdapter(
                new HibernateJpaVendorAdapter());
            return emf;
        }
    }
下一步实现配置事务管理器和在@Transactional注解的类中应用事务的切面。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@Configuration
    @EnableTransactionManagement
    public class TransactionManagersConfig {
        @Autowired
        EntityManagerFactory emf;
        @Autowired
        private DataSource dataSource;

        @Bean(name = "transactionManager")
        public PlatformTransactionManager transactionManager() {
            JpaTransactionManager tm =
                new JpaTransactionManager();
                tm.setEntityManagerFactory(emf);
                tm.setDataSource(dataSource);
            return tm;
        }
    }
注解@EnableTransactionManagement通知Spring,@Transactional注解的类被事务的切面包围。这样@Transactional就可以使用了。

总结

Spring声明式事务管理机制非常强大,但它可能被误用或者容易发生配置错误。

当这个机制不能正常工作或者未达到预期运行结果等问题出现时,理解它的内部工作情况是很有帮助的。

需要记住的最重要的一点是,要考虑到两个概念:事务和持久化上下文,每个都有自己不可读的明显的生命周期。

下一篇文章将介绍事务注解的常见陷阱以及如何避免。
分享到:
评论

相关推荐

    Spring @Transactional工作原理详解

    理解`@Transactional`的工作原理和Spring AOP代理机制,以及与EntityManager的关系,可以帮助我们更好地诊断和解决问题。 总的来说,`@Transactional`提供了声明式的事务管理,极大地简化了事务相关的代码,同时...

    Spring声明式事务@Transactional知识点分享

    "Spring声明式事务@Transactional知识点分享" 在 Spring 框架中,@Transactional 注解是实现声明式事务的关键。通过 @Transactional 注解,可以指定事务的传播行为、隔离级别、读写控制等属性。 首先,@...

    Spring中@Transactional事务回滚(含实例

    本文将深入解析`@Transactional`的事务回滚机制,并通过实例来详细讲解其工作原理,帮助读者理解和应用这一核心功能。 一、`@Transactional`注解介绍 `@Transactional`是Spring提供的一个注解,用于在方法级别或类...

    test-transactional:关于spring中@Transactional注解传播属性原理的实验

    关于spring中@Transactional注解传播属性原理的实验 具体方法: 主体形式:a方法调用b方法 a插入数据“one” b插入数据“two” a、b方法都可以有不同的传播级别或者不加事务注解(none): required(rd), required_...

    Spring中的@Transactional事物回滚实例源码

    本文将深入探讨这个注解的工作原理、如何配置以及如何在遇到异常时触发事务回滚。 首先,`@Transactional`是Spring提供的一个编程式事务管理方式,它允许我们在方法上声明事务属性,如传播行为、隔离级别、读写模式...

    Spring源码学习十二:@Transactional是如何工作的1

    Spring 框架中 @Transactional 注解的工作原理分析 在 Spring 框架中,@Transactional 注解是一个非常重要的概念,经常用于数据库操作。那么,@Transactional 注解是如何工作的呢?让我们深入源码分析。 首先,从 ...

    带有@Transactional和@Async的循环依赖问题

    总之,理解Spring的代理机制和事务、异步执行的工作原理是解决此类问题的关键。在开发过程中,应尽量避免出现复杂的循环依赖,特别是当涉及到事务和异步处理时,以确保代码的稳定性和可维护性。

    Spring Data JPA系列4——Spring声明式事务处理与多数据源支持.doc

    下面,我们将探讨这些问题,并了解 Spring 声明式事务处理机制的实现原理和多数据源支持机制。 一、事务处理机制 事务处理机制是用于确保数据库操作的一致性和可靠性的机制。它可以确保多个操作作为一个单元来执行...

    spring中注解的实现原理

    本文将深入探讨Spring中注解的实现原理,帮助你更好地理解和运用这些核心概念。 首先,让我们从注解的基础知识开始。注解在Java中是一种元数据,它提供了在代码中嵌入信息的方式,而这些信息可以被编译器或运行时...

    transaction.zip

    本篇文章将深入探讨`@Transactional`注解的使用、工作原理以及相关配置。 `@Transactional`是Spring框架提供的一个用于处理数据库事务的注解。它可以在类级别或方法级别上应用,用来指示方法应该在一个事务上下文中...

    Spring事务annotation原理详解

    Spring事务annotation原理详解 Spring框架中的事务管理是通过AOP(Aspect-Oriented Programming)来实现的,对于开发者来说,使用Spring的事务management功能,可以大大简化事务处理的复杂度。Spring的事务管理是...

    DynamicSpringTransactional:http上博客文章的支持代码

    在Spring框架中,`@Transactional`注解是用于标记事务管理的方法,它简化了数据库操作...同时,查阅Spring的官方文档,了解`@Transactional`注解的工作原理以及AOP的相关概念,这将有助于你更好地理解项目的实现细节。

    Spring事务不生效.pdf

    首先,我们要明白Spring事务的工作原理。Spring采用AOP(面向切面编程)来实现事务管理,通过动态代理在方法执行前后插入事务处理逻辑。当在服务层方法上添加@Transactional注解时,Spring会创建一个代理对象来包裹...

    Spring-Git-Mybatis-SQL性能优化-Https原理等.rar

    @Transactional实现原理 git命令应用 https原理 jsp运行原理 mybatis运行原理 Spring Boot与Spring Cloud SQL性能优化 熟悉项目 svn提交代码合并分支 查看服务器是否安装了mysql,发布应用 开发准备工作流程 学习JDK...

    spring 事务基于注解模式

    本文将深入探讨Spring中的基于注解的事务管理及其工作原理。 ### 1. Spring事务管理概述 Spring事务管理分为编程式和声明式两种。编程式事务管理通过编程的方式(如使用`TransactionTemplate`或直接调用`...

    Spring技术内幕:深入解析Spring架构与设计原理(完整版)

    在本文中,我们将详细探讨Spring架构的关键组成部分、设计原则以及它们如何协同工作,帮助开发者构建高效、可扩展的Java应用。 首先,Spring框架的核心特性之一是依赖注入(Dependency Injection, DI)。DI使得对象...

    Spring學習筆記DEMO代碼(二)

    1. Spring声明式事务管理的工作原理。 2. 如何在Spring中配置Hibernate事务管理。 3. `@Transactional`注解的使用及其属性含义。 4. 如何处理事务的异常和回滚规则。 5. 实现Spring和Hibernate的无缝集成,提升代码...

    02-01-10-Spring事务传播原理及数据库事务操作原理1

    在本课程中,我们将深入探讨Spring事务传播原理和数据库事务操作的基本概念,这对于有Spring开发经验的人员来说,是进一步深化事务控制理解的关键。我们还将触及分布式事务的初步知识,帮助开发者更好地掌握基于...

    Spring 注解学习手札(五) 业务层事务处理

    要深入了解Spring事务管理的工作原理,可以研究Spring框架的源码,重点关注`TransactionInterceptor`类,它是AOP代理创建的切面,负责处理事务的开始、提交、回滚等操作。 7. **工具支持** 在开发过程中,使用IDE...

Global site tag (gtag.js) - Google Analytics