找回密码
 立即注册
首页 业界区 业界 spring-事务管理

spring-事务管理

能杜孱 2 小时前
事务支持

什么是事务?

在一个业务流程中,需要多条DML(insert、delete、update)语句联合才能完成。这些语句必须同时成功或者同时失败。这样才能保证数据安全。
多条DML同时成功或者同时失败,叫做事务。
事务处理的四个过程


  • 开启事务
  • 执行业务代码
  • 提交事务(没出现异常,提交成功。commit transaction)
  • 回滚事务(出现异常。执行回滚事务. rollback transaction)
事务的四个特性(ACID)


  • A原子性:事务是最小的工作单元,不可分
  • C一致性:事务要么同时成功,要么同时失败
  • I隔离性:事务与事务之间保证和互不干扰
  • D持久性:持久性是事务结束的标志。
spring对事务的支持

spring实现事务的2种方式:

  • 编程式事务:通过编写代码的方式来实现事务管理
  • 声明式事务:基于注解方式和基于xml方式(推荐使用)
spring事务管理api

spring对事务的管理是基于aop实现的。所以spring专门针对事务开发了一套api,其核心接口如下:PlatformTransactionManager 接口。
声明式事务基于注解方式实现

需要配置xml文件
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3.        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  4.        xmlns:context="http://www.springframework.org/schema/context"
  5.        xmlns:tx="http://www.springframework.org/schema/tx"
  6.        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
  7.                           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
  8.                            http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd" >
  9.     <context:component-scan base-package="com.ali" />
  10.     <bean id="dataSource" >
  11.         <property name="url" value="jdbc:mysql://localhost:3306/bank"/>
  12.         <property name="username" value="root"/>
  13.         <property name="password" value="123456"/>
  14.         <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
  15.     </bean>
  16.     <bean id="jdbcTemplate" >
  17.         <property name="dataSource" ref="dataSource"/>
  18.     </bean>
  19.     <bean id="transactionManager" >
  20.         <property name="dataSource" ref="dataSource"/>
  21.     </bean>
  22.     <tx:annotation-driven transaction-manager="transactionManager"/>
  23. </beans>
复制代码
可以在类和方法上加@Transactional 开启事务

  • 加在类上表示这个类上的所有方法都开启事务
  • 加在方法方法上表示只有这个方法开启事务
事务的传播行为

什么是事务的传播行为?
在service种有a()和b()2个方法,a()上有事务,b()上也有事务,当a()在执行过程中调用了b(),事务是如何传递的?是合并到一个事务?还是开启一个新事务?这就是事务的传播行为。
一共有7种传播行为:

  • REQUIRD:支持当前事务,如果不存在就新建一个事务(默认)【没有就新建,有就加入】
  • SUPPORTS:支持当前事务,如果当前没有事务,就以非事务方式执行【有就加入,没有就不管了】
  • MANDATORY:必须运行在一个事务中,如果当前没有事务发生,将抛出异常【有就加入,没有就抛异常】
  • REQUIRES_NEW:开启一个新事务,如果一个事务已经存在,则将这个存在的事务挂起【不管有没有。直接开启一个新事务。新事务和旧事务不存在嵌套关系,旧事务被挂起了】
  • NOT_SUPPORTED:以非事务方式运行。如果有事务。则挂起当前事务【不支持事务,存在就挂起】
  • NEVER:以非事务方式运行。如果有事务。则抛异常【不支持事务,存在就抛异常】
  • NESTED:如果当前有一个事务在进行中,则该方法应当运行在一个嵌套事务中。被嵌套的事务可以独立于外层事务进行提交或回滚。如果外层事务不存在。行为就像REQUIRD一样【有事务的话,就在这个事务里嵌套一个完全独立的事务,嵌套的事务可以独立的提交和回滚。没有事务就和REQUIRD一样】
在代码中设置事务的传播行为:
  1. @Transactional(propagation = Propagation.MANDATORY)
复制代码
事务隔离级别

数据库中读取数据存在三大问题:

  • 脏读:读取到没有提交到数据库的数据
  • 不可重复读:在同一个事务中,第一次和第二次读取的数据不一样
  • 幻读:读到的数据是假的()
事务的隔离级别有4个:

  • 读未提交READ_UNCOMMITTED: 存在脏读、不可重复读、幻读问题
  • 读提交READ_COMMITTED:事务提交之后才读到。存在不可重复读、幻读问题
  • 可重复读REPEATABLE_READ:解决不可重复读的问题,存在幻读问题
  • 序列化SERIALIZABLE:解决幻读问题,事务排队执行。不支持并发。
MySQL默认可重复读,Oracle默认读提交
仅在读的事务中设置隔离级别就行,写的事务没必要设置
代码中设置事务的隔离级别:
  1. @Transactional(isolation = Isolation.DEFAULT)
复制代码
事务超时
  1. @Transactional(timeout = 10)
复制代码
以上代码设置事务超时时间为10s
表示超过10s,如果事务中所有的DML语句还没有执行完毕的话,最终结果会回滚。
默认值-1,表示没有时间限制。
事务的超时时间值得是哪段时间?
在当前事务中,最后一条DML语句执行之前的时间,如果最后一条DML语句后面有很多业务逻辑,这些业务代码执行的时间不被计入超时时间。
只读事务
  1. @Transactional(readOnly = true)
复制代码
将当前事务设为只读事务,在该事务中只允许执行select 语句。
该特性的作用是:启动spring的优化策略。提高select语句的执行效率。
设置哪些异常回滚事务
  1. @Transactional(rollbackFor = RuntimeException.class)
复制代码
表示发生RuntimeException异常或该异常的子类异常才回滚
设置哪些异常不回滚事务
  1. @Transactional(noRollbackFor = NullPointerException.class)
复制代码
表示发生NullPointerException异常或该异常的子类不回滚,其他异常才回滚
事务的全注解式开发

编写配置类
  1. @Configuration // 代替xml配置文件
  2. @ComponentScan("com.ali") // 扫描com.ali包下的所有类
  3. @EnableTransactionManagement // 开启事务管理
  4. public class Spring6Config {
  5.     // @Bean注解用于将方法的返回值注册为Spring容器中的一个Bean
  6.     @Bean(name = "druidDataSource")
  7.     public DruidDataSource druidDataSource() {
  8.         DruidDataSource druidDataSource = new DruidDataSource();
  9.         druidDataSource.setUrl("jdbc:mysql://localhost:3306/spring6?useSSL=false&serverTimezone=UTC");
  10.         druidDataSource.setUsername("root");
  11.         druidDataSource.setDriverClassName("com.mysql.jdbc.Driver");
  12.         druidDataSource.setPassword("123456");
  13.         return druidDataSource;
  14.     }
  15.     @Bean(name = "transactionManager")
  16.     public DataSourceTransactionManager transactionManager(DataSource dataSource) {
  17.         DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
  18.         transactionManager.setDataSource(dataSource);
  19.         return transactionManager;
  20.     }
  21.     @Bean(name = "jdbcTemplate")
  22.     // 该方法的参数DataSource dataSource会自动从Spring容器中找到类型为DataSource的Bean并注入
  23.     public JdbcTemplate jdbcTemplate(DataSource dataSource) {
  24.         JdbcTemplate jdbcTemplate = new JdbcTemplate();
  25.         jdbcTemplate.setDataSource(dataSource);
  26.         return jdbcTemplate;
  27.     }
  28. }
复制代码
使用时和其他方式一样。

来源:程序园用户自行投稿发布,如果侵权,请联系站长删除
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

相关推荐

您需要登录后才可以回帖 登录 | 立即注册