原文转自:获取
Spring的早期版本用户必须通过TransactionProxyFactoryBean代理对需要事务管理的业务类进行代理,便于实施事务功能的增强。
让我们先看代码吧!
业务层代码:
1
2
3
4
5
|
public interface UserScoreService {
public UserScore getUserSocore(String userNo);
public void addUserScore(UserScore us);
public int getUsableScore(String userNo);
} |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
public class UserScoreServiceImpl implements UserScoreService {
private UserScoreRepository userScoreRepository;
public void setUserScoreRepository(UserScoreRepository userScoreRepository) {
this .userScoreRepository = userScoreRepository;
}
@Override
public UserScore getUserSocore(String userNo) {
return userScoreRepository.getUserSocore(userNo);
}
@Override
public void addUserScore(UserScore us) {
userScoreRepository.addUserScore(us);
}
@Override
public int getUsableScore(String userNo) {
return userScoreRepository.getUsableScore(userNo);
}
} |
持久层代码:
1
2
3
4
5
|
public interface UserScoreRepository {
public UserScore getUserSocore(String userNo);
public void addUserScore(UserScore us);
public int getUsableScore(String userNo);
} |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
|
public class UserScoreRepositoryImpl implements UserScoreRepository {
private JdbcTemplate jdbcTemplate;
public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
this .jdbcTemplate = jdbcTemplate;
}
@Override
public UserScore getUserSocore(String userNo) {
final UserScore us = new UserScore();
StringBuffer sql = new StringBuffer();
sql.append( "select * from userscore where userNo = ?" );
jdbcTemplate.query(sql.toString(), new Object[]{userNo}, new RowCallbackHandler() {
@Override
public void processRow(ResultSet rs) throws SQLException {
us.setId(rs.getInt( "id" ));
us.setHonourScore(rs.getInt( "honourScore" ));
us.setUsableScore(rs.getInt( "usableScore" ));
us.setUserName(rs.getString( "userName" ));
us.setUserNo(rs.getString( "userNo" ));
}
});
return us;
}
@Override
public void addUserScore(UserScore us) {
}
@Override
public int getUsableScore(String userNo) {
return 0 ;
}
} |
Spring配置文件:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
<!-- 定义一个数据源 --> < bean id = "dataSource" class = "org.apache.tomcat.jdbc.pool.DataSource" >
< property name = "driverClassName" value = "com.mysql.jdbc.Driver" />
< property name = "url" value = "jdbc:mysql://localhost:3306/spring_test" />
< property name = "username" value = "root" />
< property name = "password" value = "root" />
</ bean >
<!-- 定义JdbcTemplate的Bean --> < bean id = "jdbcTemplate" class = "org.springframework.jdbc.core.JdbcTemplate"
p:dataSource-ref = "dataSource" >
</ bean >
< bean id = "userScoreRepository_jdbc"
class = "net.hingyi.springDemo.transaction.repository.UserScoreRepositoryImpl"
p:jdbcTemplate-ref = "jdbcTemplate" />
<!-- 声明事务管理器 --> < bean id = "txManager"
class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" >
< property name = "dataSource" ref = "dataSource" ></ property >
</ bean >
<!-- 需要实施事务增强的目标业务Bean --> < bean id = "userScoreTarget"
class = "net.hingyi.springDemo.transaction.service.UserScoreServiceImpl"
p:userScoreRepository-ref = "userScoreRepository_jdbc" />
<!-- 使用事务代理工厂类为目标业务Bean提供事务增强 --> <!-- p:transactionManager-ref="txManager" 指定事务管理器 --> <!-- p:target-ref="userScoreTarget" 指定目标业务Bean --> < bean id = "userScore" class = "org.springframework.transaction.interceptor.TransactionProxyFactoryBean"
p:transactionManager-ref = "txManager"
p:target-ref = "userScoreTarget" >
< property name = "transactionAttributes" >
< props >
< prop key = "get*" >PROPAGATION_REQUIRED,readOnly</ prop > <!-- 只读事务 -->
< prop key = "*" >PROPAGATION_REQUIRED</ prop > <!-- 可写事务 -->
</ props >
</ property >
</ bean >
|
web.xml配置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
< context-param >
< param-name >log4jConfigLocation</ param-name >
< param-value >classpath:log4j.properties</ param-value >
</ context-param >
< context-param >
< param-name >contextConfigLocation</ param-name >
< param-value >classpath:applicationContext.xml</ param-value >
</ context-param >
< listener >
< listener-class >org.springframework.web.util.Log4jConfigListener</ listener-class >
</ listener >
< listener >
< listener-class >org.springframework.web.context.ContextLoaderListener</ listener-class >
</ listener >
|
异常回滚/提交规则
上面spring配置中的<prop>内的值为事务属性信息,匹配格式为:
PROPAGATION (传播行为) |
, |
ISOLATION (隔离级别(可选)) |
, |
readOnly (是否为只读事务(可选)) |
, |
-Exceptions (发生这些异常时回滚事务(可选))
|
, |
+Exceptions (发生这些异常时照样提交事务(可选)) |
Spring事务的传播类型
Spring在TransactionProxyFactory接口中规定了7种类型的事务传播行为,它们规定了事务方法和事务发生嵌套调用时候是怎么传播的。如:
事务传播行为类型 |
说明 |
PROPAGATION_REQUIRED |
如果当前没有事务,就新建一个事务;如果已经存在一个事务,加入到这个事务中。(这个是最常见的选择) |
PROPAGATION_SUPPORTS |
支持当前事务。如果当前没有事务,就以非事务方式执行 |
PROPAGATION_MANDATORY |
使用当前的事务。如果当前没有事务,就抛出异常 |
PROPAGATION_REQUIRES_NEW |
新建事务。如果当前存在事务,就把当前事务挂起 |
PROPAGATION_NOT_SUPPORTED |
以非事务的方式执行操作。如果当前存在事务,就把当前事务挂起 |
PROPAGATION_NEVER |
以非事务方式执行。如果当前存在事务,则抛出异常 |
PROPAGATION_NESTED |
如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与 PROPAGATION_REQUIRED类似的操作 |
隔离级别
隔离级别是可选的,默认的为ISOLATION_DEFAULT,表示数据库的默认隔离级别。隔离级别的值如下:
隔离级别 |
脏读 |
不可重复读 |
幻象读 |
第一类丢失更新 |
第二类丢失更新 |
ISOLATION_READ_UNCOMMITED |
允许 |
允许 |
允许 |
不允许 |
允许 |
ISOLATION_READ_COMMITED |
不允许 |
允许 |
允许 |
不允许 |
允许 |
ISOLATION_REPEATABLE_READ |
不允许 |
不允许 |
允许 |
不允许 |
不允许 |
ISOLATION_SERIALIZABLE |
不允许 |
不允许 |
不允许 |
不允许 |
不允许 |
默认情况下,当发生运行异常时,事务将被回滚,发生检查型异常时,既不回滚也不提交,控制权交给外层调用。所以,带负号的异常设置仅对检查型异常有意义。
相关推荐
在Spring框架中,事务管理是核心特性之一,它允许开发者以声明式或编程式的方式处理应用中的事务。Spring事务管理的目的是确保数据的一致性和完整性,尤其是在多操作、多资源的环境中。本Demo将深入探讨Spring如何...
Spring提供两种事务管理方式:编程式事务管理和声明式事务管理。编程式事务管理通过PlatformTransactionManager接口实现,代码中显式地开始、提交或回滚事务。声明式事务管理则通过@Transactional注解,将事务管理...
spring事务控制jar包,请网上自寻下载
**声明式事务管理** 是通过在方法上添加`@Transactional`注解,让Spring自动管理事务。这种方式更加简洁,降低了代码的复杂性。`@Transactional`可以设置不同的属性,如`propagation`(传播行为)、`isolation`...
同时,Spring还提供了编程式事务管理,允许开发者在代码中手动管理事务,但这种方式通常在更复杂的场景或特殊需求下使用。 通过这个实验,学生可以深入理解Spring声明式事务管理的工作原理,以及如何在实际项目中...
3. 由Spring管理事务生命周期:Spring会在方法执行前后自动开始和结束事务,如果发生异常,Spring会自动回滚事务,否则在方法正常结束时提交事务。 四、Spring事务注解管理 在Spring中,可以使用@Transactional...
然而,在Spring中,通常推荐使用Spring管理事务,因为这样可以更好地进行事务的统一控制和事务的声明式管理。 1. **Hibernate事务管理:** - Session API:Hibernate的Session对象是操作数据库的基本单元,它支持...
在本文中,我们将深入探讨Spring框架中的事务管理。Spring是一个广泛应用的Java企业级应用开发框架,它提供了强大的事务管理功能,使得开发者可以方便地控制事务的边界,保证数据的一致性和完整性。 首先,理解事务...
编程式事务管理是指通过编程的方式来管理事务,而声明式事务管理是指通过配置的方式来管理事务。 事务管理的隔离级别 Spring 中的事务管理提供了多种隔离级别,包括: * ISOLATION_DEFAULT:使用数据库的默认隔离...
编程式事务管理通过TransactionTemplate或直接调用PlatformTransactionManager接口的方法来管理事务,而声明式事务管理则通过在配置文件或注解中定义事务规则,让Spring自动处理事务开始、提交、回滚等操作。...
### 深入理解Spring的事务管理机制 #### 一、事务的基本原理 Spring框架的事务管理机制是在Java开发环境中非常重要的一个组成部分,它能够帮助开发者简化事务处理的复杂度,提高应用程序的一致性和可靠性。Spring...
在需要事务控制的方法上添加这个注解,Spring会自动管理事务的开始、提交、回滚等操作。 2. 编程式事务管理: 编程式事务管理则是在代码中直接控制事务的开始、提交、回滚等操作,通常通过TransactionTemplate或...
### Spring如何管理事务 #### 一、Spring事务管理概述 Spring框架为开发者提供了一套强大的事务管理机制,它简化了应用程序中的事务控制逻辑,使得开发人员能够更加专注于业务逻辑的编写,而不是繁琐的事务管理...
- **配置事务管理器**:接下来,需要配置一个事务管理器(如`DataSourceTransactionManager`),它负责管理事务的开始、提交和回滚。 - **配置事务规则**:最后,需要配置事务规则,指定哪些方法需要在事务中执行。 ...
Spring 提供了PlatformTransactionManager接口,如DataSourceTransactionManager和JtaTransactionManager,用于管理事务。开发人员可以通过TransactionTemplate或直接使用TransactionStatus接口来控制事务的生命周期...
在Spring中,可以通过JDBC驱动与MySQL交互,并由Spring管理事务。 **Maven** 是一个项目管理和综合工具,它可以帮助开发者管理项目的构建、依赖和文档生成。在本案例中,Maven可能被用来下载和管理所需的库,编译源...
在Spring中,我们可以配置`PlatformTransactionManager`接口的实现类,如`HibernateTransactionManager`,它会自动感知Hibernate Session并管理事务。通过声明式事务管理,我们只需在方法上添加`@Transactional`注解...
3. **TransactionProxyFactoryBean** 或者 `@EnableTransactionManagement`:用于创建代理对象,拦截带有@Transactional的方法并管理事务。 在源码分析中,我们可以看到当一个方法被@Transactional标记后,Spring会...
Spring 事务管理是Java开发中一个非常重要的概念,特别是在企业级应用开发中,它确保了数据的一致性和完整性。在Spring框架中,事务管理分为编程式事务管理和声明式事务管理两种方式。 一、编程式事务管理 编程式...