Spring事务的七种传播行为
文章目录
1、环境搭建
1.1 sql
CREATE TABLE `student` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '主键',
`name` varchar(255) DEFAULT NULL COMMENT '姓名',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=162 DEFAULT CHARSET=utf8;
1.2 pom
<dependencies>
<!-- spring框架运行的必须jar包 -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<!-- spring框架对于jdbc进行的简单封装 -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<!-- spring框架对于事务支持的jar包 -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<!-- spring框架的测试包 -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-test</artifactId>
<version>5.0.2.RELEASE</version>
</dependency>
<!-- spring框架提供对AspectJ框架的整合 -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.7</version>
</dependency>
<!-- mysql数据库 -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.6</version>
</dependency>
<!-- 测试 -->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.16.4</version>
</dependency>
</dependencies>
1.3 配置文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置spring提供的数据库连接池 -->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
<property name="url" value="jdbc:mysql://localhost:3306/spring"></property>
<property name="username" value="root"></property>
<property name="password" value="123456"></property>
</bean>
<!-- jdbcTemplate的配置 -->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"></property>
</bean>
<context:component-scan
base-package="com"></context:component-scan>
<!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 启用事务注解 -->
<tx:annotation-driven transaction-manager="transactionManager"/>
</beans>
1.4 码云地址
https://gitee.com/baizeze/propagation.git
2、传播行为
2.1 PropagationRequired
spring默认的传播行为,如果外围方法有事务 则使用外围方法事务 如果外围没有事务 该传播行为基本就事务失效了
@Service
public class PropagationRequiredService {
@Autowired
private PropagationRequiredService propagationRequiredService;
/**
* spring默认的传播行为,如果外围方法有事务 则使用外围方法事务 如果外围没有事务 该传播行为基本就事务失效了
*/
@Autowired
private JdbcTemplate jdbcTemplate;
@Transactional(propagation = Propagation.REQUIRED)
public void add(Student student){
jdbcTemplate.update("insert into student (name) values (?);",
student.getName());
}
@Transactional(propagation = Propagation.REQUIRED)
public void update(Student student){
//这种写法的目的是为了防止事务失效 但是默认的传播行为
//add(student) 这种写法在等价于没有注解 直接把代码写在update方法体里
propagationRequiredService.add(student);
student.setName("update");
jdbcTemplate.update("UPDATE student SET NAME =?",
student.getName());
}
}
2.2 PropagationSupports
如果外围方法有事务就使用外围方法的事务 如果没有就按照非事务去运行
@Service
public class PropagationSupportsService {
@Autowired
private PropagationSupportsService propagationSupportsService;
/**
* Supports 如果有事务就使用事务 如果没有就按照非事务运行
*/
@Autowired
private JdbcTemplate jdbcTemplate;
/**
* 这种场景事务是失效的
* @param student
*/
@Transactional(propagation = Propagation.SUPPORTS)
public void add(Student student){
jdbcTemplate.update("insert into student (name) values (?);",
student.getName());
}
@Transactional(propagation = Propagation.REQUIRED)
public void update(Student student){
//add(student) 和43行一样也是使用一个事务
propagationSupportsService.add(student);
student.setName("update");
jdbcTemplate.update("UPDATE student SET NAME =?",
student.getName());
}
2.3PropagationMandaory
该传播行为是如果没有事务 就会抛异常No existing transaction found for transaction marked with propagation ‘mandatory’
@Service
public class PropagationMondatoryService {
/**
* 外围有事务则使用事务 没有事务就抛出异常
* No existing transaction found for transaction marked with propagation 'mandatory'
*/
@Autowired
private PropagationMondatoryService propagationMondatoryService;
@Autowired
private JdbcTemplate jdbcTemplate;
@Transactional(propagation = Propagation.MANDATORY)
public void add(Student student){
jdbcTemplate.update("insert into student (name) values (?);",
student.getName());
}
@Transactional(propagation = Propagation.SUPPORTS)
public void update(Student student){
//add() 注解失效 无法验证
//这种方法可以验证
propagationMondatoryService.add(student);
student.setName("update");
jdbcTemplate.update("UPDATE student SET NAME =?",
student.getName());
}
}
2.4 PropagationRequiedNew
该传播行为如果外围有事务就把外围事务挂起 重新开启一个新的事务 外围事务异常不会影响内部事务提交
@Service
public class PropagationRequiredNewService {
/**
* 如果外围有事务 重新开启一个事务 暂时挂起外围事务 外围事务异常不会影响内部事务的提交
*
*/
@Autowired
private PropagationRequiredNewService propagationRequiredNewService;
@Autowired
private JdbcTemplate jdbcTemplate;
/**
* 自身是支持事务的
* @param student
*/
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void add(Student student){
jdbcTemplate.update("insert into student (name) values (?);",
student.getName());
//int i=1/0;
}
@Transactional(propagation = Propagation.REQUIRED)
public void update(Student student){
//add() 注解失效 无法验证
propagationRequiredNewService. add(student);
student.setName("REQUIRES_NEW");
jdbcTemplate.update("UPDATE student SET NAME =?",
student.getName());
int i=1/0;
}
}
2.5 PropagationNotSupported
该传播行为总是已非事务运行
@Service
public class PropagationNotSupportedService {
/**
* 总是以非事务运行
*/
@Autowired
private PropagationNotSupportedService propagationNotSupportedService;
@Autowired
private JdbcTemplate jdbcTemplate;
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void add(Student student){
jdbcTemplate.update("insert into student (name) values (?);",
student.getName());
}
@Transactional(propagation = Propagation.REQUIRED)
public void update(Student student){
//add() 注解失效 无法验证
//这种方法可以验证
propagationNotSupportedService.add(student);
student.setName("update");
jdbcTemplate.update("UPDATE student SET NAME =?",
student.getName());
int i=1/0;
}
}
2.6 PropagationNever
该传播行为是指存在事务就跑异常Existing transaction found for transaction marked with propagation ‘never’
@Service
public class PropagationNeverService {
/**
* 总是以非事务运行 如果有事务则抛异常
* Existing transaction found for transaction marked with propagation 'never'
*/
@Autowired
private PropagationNeverService PropagationNeverService;
@Autowired
private JdbcTemplate jdbcTemplate;
@Transactional(propagation = Propagation.NEVER)
public void add(Student student){
jdbcTemplate.update("insert into student (name) values (?);",
student.getName());
}
@Transactional(propagation = Propagation.REQUIRED)
public void update(Student student){
//add() 注解失效 无法验证
//这种方法可以验证
PropagationNeverService.add(student);
student.setName("update");
jdbcTemplate.update("UPDATE student SET NAME =?",
student.getName());
}
}
2.7PropagationNested
如果存在事务 改方法会嵌套在该事务的子事务中,子事务的提交和回滚 可以被主事务决定 子事务的异常不会影响主事务的(参考文章总结 目前没有接触过类似的代码)
3 、总结
传播行为可以分为以下四类
- 不支持事务 PropagationSuppotrs PropagetionNotSupport
- 事务异常 PropagationMandatory ProPagationNever
- 支持事务 PropagationRequired
- 相互影响 PropagationRequiredNew PropagationNested