[Spring框架]Spring 事务管理基础入门总结.

时间:2023-03-09 01:17:09
[Spring框架]Spring 事务管理基础入门总结.

前言:
在之前的博客中已经说过了数据库的事务, 不过那里面更多的是说明事务的一些锁机制, 今天来说一下Spring管理事务的一些基础知识. 
之前的文章: [数据库事务与锁]详解一: 彻底理解数据库事务一, 什么是事务

事务是逻辑上一组操作,这组操作要么全都成功,要么全都失败.

事务的属性: ACID
原子性(Atomicity): 事务作为一个整体被执行,包含在其中的对数据的操作要么全部被执行,要么都不执行.
一致性(Consistency):事务应确保数据库的状态从一个一致状态转变为另一个一致状态. 一致状态的含义是数据库中的数据应满足为完整性约束.
隔离性(Isolation):多个事务并发执行时, 一个事务的执行不应该影响其他事务的执行.
持久性(Durability):一个事务一旦提交, 他对数据库的修改应该永久保存在数据库中.
  

二, Spring的事务管理

一类:编程式事务管理.手动编写代码管理事务.

二类:声明式事务管理.通过配置完成事务管理.(AOP) : 这里只说声明式的事务管理.

事务管理的API:

PlatformTransactionManager :平台事务管理器

  * 是真正管理事务的对象.

TransactionDefinition :事务定义信息.

  * 定义事务的(隔离级别,传播行为,超时信息,只读)

TransactionStatus :事务状态.

  * 管理事务的过程中,事务有一些状态的改变.状态信息被记录在该对象中.

Spring根据TransactionDefinition中定义的信息使用PlatformTransactionManager管理事务,管理事务过程中产生状态,将状态记录到TransactionStatus中.

 

三, Spring相关API详解
1,
PlatformTransactionManager
[Spring框架]Spring 事务管理基础入门总结.

2,TransactionDefinition 

常量:

  * ISOLATION_*:定义事务的隔离级别.
  [Spring框架]Spring 事务管理基础入门总结.

  * PROPAGATION_*:定义事务的传播行为.
  [Spring框架]Spring 事务管理基础入门总结.
  
* 超时信息:
  [Spring框架]Spring 事务管理基础入门总结.
方法:

  [Spring框架]Spring 事务管理基础入门总结.

Transaction事务状态:
  [Spring框架]Spring 事务管理基础入门总结.|

Spring的事务传播行为:

PROPAGATION_REQUIRED :如果A,B操作,如果A有事务使用A的事务将B包含进来.如果A没有事务创建一个新事务将A,B包进来.
PROPAGATION_SUPPORTS :如果A,B操作,如果A有事务,使用A的事务将B包含进来.如果A没有事务,不使用事务.
PROPAGATION_MANDATORY :如果A,B操作,如果A有事务,使用A的事务将B包含进来.如果A没有事务,就会抛出异常. PROPAGATION_REQUIRES_NEW :如果A,B操作,如果A有事务,将A的事务挂起.创建一个新事务执行B操作.
PROPAGATION_NOT_SUPPORTED :如果A,B操作,总是以非事务方式运行,如果A有,将A挂起.
PROPAGATION_NEVER :如果A,B操作,总是以非事务方式运行,如果A有事务抛出异常. PROPAGATION_NESTED :嵌套事务.如果A,B操作,A有事务,在A执行完之后设置一个保存点.如果B执行没有问题,一起提交.如果B出现问题.允许用户自己控制回滚到最初还是回滚到保存点.

四, Spring事务管理实例(声明式事务管理)

Spring声明式事务管理又可分为三种:
  1,
基于TransactionProxyFactoryBean
   2, 基于AspectJ的XML方式的事务管理
  3, 基于AspectJ的注解方式事务管理

基于AspectJ的XML方式的事务管理:
  1,引入开发包:
  [Spring框架]Spring 事务管理基础入门总结.

  2,Service层代码:
AccountService.java

 public interface AccountService {

     /**
* 转账的方法
*/
public void transfer(String from,String to,Double money);
}

AccountServiceImpl.java:

     private AccountDao accountDao;

     public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
} @Override
/**
* 业务层转账的方法:
* from:转出账号
* to:转入账号
* money:转账金额
*/
public void transfer(final String from, final String to, final Double money) {
accountDao.outMoney(from, money);
int d = 1 / 0;
accountDao.inMoney(to, money);
} }

  3,DAO层代码
AccountDAO.java:

 public interface AccountDao {
public void outMoney(String from, Double money); public void inMoney(String to, Double money);
}

AccountDAOImpl.java:

 public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {

     @Override
public void outMoney(String from, Double money) {
this.getJdbcTemplate().update("update account set money = money - ? where name = ?", money,from);
} @Override
public void inMoney(String to, Double money) {
this.getJdbcTemplate().update("update account set money = money + ? where name = ?", money,to);
} }

  4,配置文件
jdbc.properties:

jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql:///spring_transaction
jdbc.user=root
jdbc.password=123

applicationcontext.xml:

 <?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:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd"> <context:property-placeholder location="classpath:jdbc.properties"/> <!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}"/>
<property name="jdbcUrl" value="${jdbc.url}"/>
<property name="user" value="${jdbc.user}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<!-- 配置Service -->
<bean id="accountService" class="cn.augmentum.transaction.demo3.service.AccountServiceImpl">
<property name="accountDao" ref="accountDao"/>
</bean> <!-- 配置DAO -->
<bean id="accountDao" class="cn.augmentum.transaction.demo3.dao.AccountDaoImpl">
<property name="dataSource" ref="dataSource"/>
</bean> <!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean> <!-- 配置事务增强 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<!--
propagation:事务传播行为:
isolation:隔离级别
read-only:是否为只读事务.
timeout :是否超时.
rollback-for:类似-Exception.发生哪些异常回滚事务.
no-rollback-for:类似+Exception.发生哪些异常不回滚.
-->
<tx:method name="transfer" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice> <!-- AOP的配置 -->
<aop:config>
<aop:pointcut expression="execution(* cn.augmentum.transaction.demo3.service.AccountServiceImpl.*(..))" id="pointcut1"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref=""/>
</aop:config>
</beans>

  5, 测试方法
SpringDemo.java:

 @RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class SpringDemo3 { @Resource(name="accountService")
private AccountService accountService; @Test
public void demo1(){
accountService.transfer("老马", "凤姐", 1000d);
}
}

关于XML的声明式事务管理就这些了, 那么下面来看下更简单的关于注解的声明式事务管理的做法.;
这里只需要看Service层是如何加注解以及applicationcontext中是怎样配置的就行了.

AccountServiceImpl.java:

 /**
* @Transactional中属性
* propagation:传播行为.
* readOnly:是否只读
* timeout:超时信息
* isolation:隔离级别
*/
@Transactional(propagation=Propagation.REQUIRED,readOnly=false,timeout=-1,isolation=Isolation.DEFAULT)
public class AccountServiceImpl implements AccountService {
private AccountDao accountDao; public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
} @Override
/**
* 业务层转账的方法:
* from:转出账号
* to:转入账号
* money:转账金额
*/
public void transfer(final String from, final String to, final Double money) {
accountDao.outMoney(from, money);
int d = 1 / 0;
accountDao.inMoney(to, money);
}
}

applicationcontext.xml:

 <?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:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd"> <context:property-placeholder location="classpath:jdbc.properties"/> <!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="${jdbc.driverClass}"/>
<property name="jdbcUrl" value="${jdbc.url}"/>
<property name="user" value="${jdbc.user}"/>
<property name="password" value="${jdbc.password}"/>
</bean>
<!-- 配置Service -->
<bean id="accountService" class="cn.augmentum.transaction.demo4.service.AccountServiceImpl">
<property name="accountDao" ref="accountDao"/>
</bean> <!-- 配置DAO -->
<bean id="accountDao" class="cn.augmentum.transaction.demo4.dao.AccountDaoImpl">
<property name="dataSource" ref="dataSource"/>
</bean> <!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean> <!-- 开启注解事务 -->
<tx:annotation-driven transaction-manager="transactionManager"/>
</beans>

注解方式的声明式事务管理也就是这么多, 看起来确实简单了很多.