Spring事务管理(一)快速入门

简介: Spring事务管理(一)快速入门

先来讲讲 脏读不可重复读幻读


  1. 脏读:我们在并发编程中是很熟悉的,通俗的讲就是你读得数据已经被修改了,已经过时失去意义了。


  1. 不可重复读: 同一个事务里面多次读取同一行数据,却返回不同的结果。


  1. 幻读:同样一笔查询在整个事务过程中多次执行后,查询所得的结果集不一样。


事务四大特性 ACID


1. 原子性 (Atomicity)


要求事务所包含的全部操作是一个不可分割的整体,如果有一步发生异常,则全部不提交。


2. 一致性 Consistency


A给B转钱,A减和B增这两个操作必须保持一致。


3. 隔离性 Isolation


事务会将一部分数据与其他事务隔离,防止脏读等。


4. 持久性 Durability


事务的结果被写到持久化存储器中。


事务四大隔离级别


隔离级别越高,则性能相对越低,反之亦然。


1. Read Uncommitted


最低的隔离级别,跟你直译的意思一样:可以读取其它事务未完成的结果。(脏读)

很明显, 脏读不可重复读幻读这三个问题它都有。


2. Read Committed


大部分数据库采用的默认隔离级别,比上一个隔离级别多了限定:在该事务完成后,才能读取该事务的数据更新后的结果。


它可以避免脏读,但是也有不可重复读取和幻读的问题。


3. Repeatable Read


可以保证在整个事务的过程中,对同一笔数据的读取结果是相同的,不管其他事务是否同时在对同一笔数据进行更新,也不管其他事务对同一笔数 据的更新提交与否。


Repeatable Read隔离级别避免了脏读和不可重复读取的问题,但无法避免幻读。


4. Serializable


最为严格的隔离级别,所有的事务操作都必须依次顺序执行,可以避免其他隔离级别遇到的所有问题,是最为安全的隔离级别, 但同时也是性能最差的隔离级别。


通常情况下,我们会使用其他隔离级别加上相应的并发锁的机制来控制对数据的访问,这样既保证 了系统性能不会损失太大,也能够一定程度上保证数据的一致性。


Spring事务传播机制


image.png


从JDBC的事务说起


我们都知道,JDBC给我们提供了事务。


  1. try{
  2.     con.setAutoCommit(false);//开启事务
  3.     ......
  4.     con.commit();//try的最后提交事务      
  5. }catch(){
  6.    con.rollback();//回滚事务
  7. }


获取事务隔离级别

  1. Connection.getTransactionIsolation()


设置事务隔离级别

  1. con.setTransactionIsolation(Connection.TRANSACTION_REPEATABLE_READ);


Spring事务机制


Spring并不会直接管理事务,而是提供了事务管理器,将事务管理的职责委托给JPA JDBC JTA DataSourceTransaction JMSTransactionManager 等框架提供的事务来实现。



那么,Spring提供的事务管理器是什么呢?


PlatformTransactionManager.java接口:


PlatformTransactionManager.java


Spring提供的事务管理器。不同的事务遵循该事务管理器的API,便能很轻松的交给Spring管理。


  1. publicinterfacePlatformTransactionManager{
  2.    // 通过Transation定义 获取Transation
  3.    TransactionStatus getTransaction(@NullableTransactionDefinition var1)throwsTransactionException;
  4.    // 提交事务
  5.    void commit(TransactionStatus var1)throwsTransactionException;
  6.    // 回滚事务
  7.    void rollback(TransactionStatus var1)throwsTransactionException;
  8. }


可以看到它里面引用到了 TransactionDefinitionTransactionStatus.


TransactionDefinition.java


它里面包含了事务的定义。


  1. publicinterfaceTransactionDefinition{
  2.    // 传播机制
  3.    int PROPAGATION_REQUIRED =0;
  4.    int PROPAGATION_SUPPORTS =1;
  5.    int PROPAGATION_MANDATORY =2;
  6.    int PROPAGATION_REQUIRES_NEW =3;
  7.    int PROPAGATION_NOT_SUPPORTED =4;
  8.    int PROPAGATION_NEVER =5;
  9.    int PROPAGATION_NESTED =6;
  10.    // 隔离级别
  11.    int ISOLATION_DEFAULT =-1;
  12.    int ISOLATION_READ_UNCOMMITTED =1;
  13.    int ISOLATION_READ_COMMITTED =2;
  14.    int ISOLATION_REPEATABLE_READ =4;
  15.    int ISOLATION_SERIALIZABLE =8;
  16.    int TIMEOUT_DEFAULT =-1;

  17.    int getPropagationBehavior();
  18.    // 获取隔离级别
  19.    int getIsolationLevel();

  20.    int getTimeout();

  21.    boolean isReadOnly();

  22.    @Nullable
  23.    String getName();
  24. }

TransactionStatus.java


事务的状态。


  1. publicinterfaceTransactionStatusextendsSavepointManager,Flushable{
  2.    boolean isNewTransaction();

  3.    boolean hasSavepoint();

  4.    void setRollbackOnly();

  5.    boolean isRollbackOnly();

  6.    void flush();

  7.    boolean isCompleted();
  8. }


Spring默认事务使用


1. 代码方式使用


  1. @Autowired
  2. privatePlatformTransactionManager transactionManager;
  3. publicvoid testTX(){
  4.    DefaultTransactionDefinition definition =newDefaultTransactionDefinition();
  5.    TransactionStatus status = transactionManager.getTransaction(definition);
  6.    try{
  7.        // 业务逻辑
  8.        // ...

  9.        // 提交事务
  10.        transactionManager.commit(status);
  11.    }catch(Exception e){
  12.        // 发生异常,事务回滚
  13.        transactionManager.rollback(status);
  14.    }
  15. }

2. 注解方式使用


  1. @Transactional
  2. void testTX2(){
  3.    // 业务逻辑 ...
  4. }


这不是玄学,它的底层是依靠AOP动态代理实现,其实重新渲染出的代码和第一个使用方式类似,不过大大减少了开发复杂度。


扩展:@Transactional注解


  1. @Target({ElementType.METHOD,ElementType.TYPE})
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Inherited
  4. @Documented
  5. public@interfaceTransactional{

  6.    //指定使用的事务管理器
  7.    @AliasFor("transactionManager")
  8.    String value()default"";

  9.    @AliasFor("value")
  10.    String transactionManager()default"";
  11.    // 可选的事务传播行为设置
  12.    Propagation propagation()defaultPropagation.REQUIRED;
  13.    // 可选的事务隔离级别设置
  14.    Isolation isolation()defaultIsolation.DEFAULT;
  15.    // 事务超时时间设置
  16.    int timeout()default-1;
  17.    // 读写或只读事务,默认读写
  18.    boolean readOnly()defaultfalse;
  19.    // 导致事务回滚的异常类数组
  20.    Class<?extendsThrowable>[] rollbackFor()default{};
  21.    // 导致事务回滚的异常类名字数组
  22.    String[] rollbackForClassName()default{};
  23.    // 不会导致事务回滚的异常类数组
  24.    Class<?extendsThrowable>[] noRollbackFor()default{};
  25.    // 不会导致事务回滚的异常类名字数组
  26.    String[] noRollbackForClassName()default{};
  27. }


Spring事务实践


非入门选手下面的demo可能会引起你的不适(浪费时间)。 假设我要完成一个功能,当删除用户的时候,将与该用户有关的所有数据行都删除。


  1. publicvoid delUser(Integer userId){
  2.    // 删除和用户相关的信息
  3.     otherRepository.deleteByUserId(userId);
  4.    // 删除用户
  5.     userRepository.deleteById(userId);
  6. }


这样的写法一般来讲,会成功的完成任务。但是如果这样一段代码:


  1. publicvoid delUser(Integer userId){
  2.    // 删除和用户相关的信息
  3.    otherRepository.deleteByUserId();
  4.    if(true){
  5.        thrownewRuntimeException("xxx");
  6.    }
  7.    // 删除用户
  8.     userRepository.deleteById(userId);
  9. }


结果会是: deleteByUserId()执行成功, deleteById()执行失败,不满足数据的一致性。


所以我们需要事务来限制:要么全部执行,要么全部不执行(方法中有异常就自动回滚)。那怎么实现呢,只需要在方法上加一个注解: @Transactional


  1. @Transactional
  2. publicvoid delUser(Integer userId){
  3.    // 删除和用户相关的信息
  4.    otherRepository.deleteByUserId();
  5.    if(true){
  6.        thrownewRuntimeException("xxx");
  7.    }
  8.    // 删除用户
  9.     userRepository.deleteById(userId);
  10. }


Spring 加载第三方事务管理


比如我有个需求(接着上次的强票系统II),要求信息不能丢失,要用到RabbitMQ的事务管理,那怎么去加载到Spring的事务管理器中呢?


  1. @Bean
  2. publicConnectionFactory connectionFactory(){
  3.    CachingConnectionFactory connectionFactory =newCachingConnectionFactory();
  4.    return connectionFactory;
  5. }

  6. @Bean
  7. publicRabbitTransactionManager rabbitTransactionManager(ConnectionFactory connectionFactory){
  8.    returnnewRabbitTransactionManager(connectionFactory);
  9. }


我们只需要这样做便可以使的使用 @Transactional注解来实现对RabbitMQ的事务管理,其它框架也类似。

相关文章
|
2天前
|
SQL Java 关系型数据库
【SpringFramework】Spring事务
本文简述Spring中数据库及事务相关衍伸知识点。
26 9
|
9天前
|
Java 开发者 Spring
理解和解决Spring框架中的事务自调用问题
事务自调用问题是由于 Spring AOP 代理机制引起的,当方法在同一个类内部自调用时,事务注解将失效。通过使用代理对象调用、将事务逻辑分离到不同类中或使用 AspectJ 模式,可以有效解决这一问题。理解和解决这一问题,对于保证 Spring 应用中的事务管理正确性至关重要。掌握这些技巧,可以提高开发效率和代码的健壮性。
38 13
|
1月前
|
缓存 安全 Java
Spring高手之路26——全方位掌握事务监听器
本文深入探讨了Spring事务监听器的设计与实现,包括通过TransactionSynchronization接口和@TransactionalEventListener注解实现事务监听器的方法,并通过实例详细展示了如何在事务生命周期的不同阶段执行自定义逻辑,提供了实际应用场景中的最佳实践。
47 2
Spring高手之路26——全方位掌握事务监听器
|
2月前
|
XML Java 数据库连接
Spring高手之路25——深入解析事务管理的切面本质
本篇文章将带你深入解析Spring事务管理的切面本质,通过AOP手动实现 @Transactional 基本功能,并探讨PlatformTransactionManager的设计和事务拦截器TransactionInterceptor的工作原理,结合时序图详细展示事务管理流程,最后引导分析 @Transactional 的代理机制源码,帮助你全面掌握Spring事务管理。
39 2
Spring高手之路25——深入解析事务管理的切面本质
|
1月前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
2月前
|
Java 开发者 Spring
Spring高手之路24——事务类型及传播行为实战指南
本篇文章深入探讨了Spring中的事务管理,特别是事务传播行为(如REQUIRES_NEW和NESTED)的应用与区别。通过详实的示例和优化的时序图,全面解析如何在实际项目中使用这些高级事务控制技巧,以提升开发者的Spring事务管理能力。
63 1
Spring高手之路24——事务类型及传播行为实战指南
|
2月前
|
JavaScript Java 关系型数据库
Spring事务失效的8种场景
本文总结了使用 @Transactional 注解时事务可能失效的几种情况,包括数据库引擎不支持事务、类未被 Spring 管理、方法非 public、自身调用、未配置事务管理器、设置为不支持事务、异常未抛出及异常类型不匹配等。针对这些情况,文章提供了相应的解决建议,帮助开发者排查和解决事务不生效的问题。
|
2月前
|
XML Java 数据库连接
Spring中的事务是如何实现的
Spring中的事务管理机制通过一系列强大的功能和灵活的配置选项,为开发者提供了高效且可靠的事务处理手段。无论是通过注解还是AOP配置,Spring都能轻松实现复杂的事务管理需求。掌握这些工具和最佳实践,能
76 3
|
4月前
|
Java 数据库连接 数据库
spring复习05,spring整合mybatis,声明式事务
这篇文章详细介绍了如何在Spring框架中整合MyBatis以及如何配置声明式事务。主要内容包括:在Maven项目中添加依赖、创建实体类和Mapper接口、配置MyBatis核心配置文件和映射文件、配置数据源、创建sqlSessionFactory和sqlSessionTemplate、实现Mapper接口、配置声明式事务以及测试使用。此外,还解释了声明式事务的传播行为、隔离级别、只读提示和事务超时期间等概念。
spring复习05,spring整合mybatis,声明式事务
|
4月前
|
Java 测试技术 数据库
Spring事务传播机制(最全示例)
在使用Spring框架进行开发时,`service`层的方法通常带有事务。本文详细探讨了Spring事务在多个方法间的传播机制,主要包括7种传播类型:`REQUIRED`、`SUPPORTS`、`MANDATORY`、`REQUIRES_NEW`、`NOT_SUPPORTED`、`NEVER` 和 `NESTED`。通过示例代码和数据库插入测试,逐一展示了每种类型的运作方式。例如,`REQUIRED`表示如果当前存在事务则加入该事务,否则创建新事务;`SUPPORTS`表示如果当前存在事务则加入,否则以非事务方式执行;`MANDATORY`表示必须在现有事务中运行,否则抛出异常;
202 4
Spring事务传播机制(最全示例)

热门文章

最新文章