Spring中事务管理的四种方法(银行转账为例)

时间:2022-09-07 14:00:17

前言

本文配套示例代码下载地址(完整可运行,含sql文件,下载后请修改数据库配置):点击这里下载

一、事务的作用

 

  将若干的数据库操作作为一个整体控制,一起成功或一起失败。

  原子性:指事务是一个不可分割的工作单位,事务中的操作要么都发生,要么都不发生。

  一致性:指事务前后数据的完整性必须保持一致。

  隔离性:指多个用户并发访问数据库时,一个用户的事务不能被其他用户的事务所干扰,多个并发事务之间数据要相互隔离。

  持久性:指一个事务一旦被提交,它对数据库中数据的改变就是永久性的,即时数据库发生故障也不应该对其有任何影响。

二、spring事务管理高层抽象主要包括3个接口

 

  --platform transactionmanager 事务管理器(提交、回滚事务)

     spring为不同的持久化框架提供了不同的platform transactionmanager接口实现。如:

        使用spring jdbc或ibatis进行持久化数据时使用datasourcetransactionmanager

        使用hibernate3.0版本进行持久化数据时使用hibernatetransactionmanager

  --transactiondefinition 事务定义信息(隔离、传播、超时、只读)

        脏读:一个事务读取了另一个事务改写但还未提交的数据,如果这些数据被回滚,则读到的数据是无效的。

        不可重复读:在同一事务中,多次读取同一数据返回的结果有所不同。

        幻读:一个事务读取了几行记录后,另一个事务插入一些记录,幻读就发生了。再后来的查询中,第一个事务就会发现有些原来没有的记录。

        事务隔离级别:(五种)

  •     default--使用后端数据库默认的隔离级别(spring中的选择项)
  •     read_uncommited--允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读
  •     read_committed--允许在并发事务已经提交后读取。可防止脏读,但幻读和不可重复读仍可发生
  •     repeatable_read--对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生
  •     serializable--完全服从acid的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的

    其中,mysql默认采用repeatable_read隔离级别;oracle默认采用read_committed隔离级别

        事务传播行为:(七种)

  •     required--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
  •     supports--支持当前事务,如果当前没有事务,就以非事务方式执行。
  •     mandatory--支持当前事务,如果当前没有事务,就抛出异常。
  •     requires_new--新建事务,如果当前存在事务,把当前事务挂起。
  •     not_supported--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  •     never--以非事务方式执行,如果当前存在事务,则抛出异常。
  •     nested--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与required类似的操作。拥有多个可以回滚的保存点,内部回滚不会对外部事务产生影响。只对datasourcetransactionmanager有效

  --transactionstatus 事务具体运行状态

三、spring提供了以下方法控制事务

 

  a.编程式事务管理(基于java编程控制,很少使用)--见demo1包

       利用transactiontemplate将多个dao操作封装起来

  *b.声明式事务管理(基于spring的aop配置控制)

       -基于transactionproxyfactorybean的方式.(很少使用)--见demo2包

            需要为每个进行事务管理的类,配置一个transactionproxyfactorybean进行增强.

       -基于xml配置(经常使用)--见demo3包

            一旦配置好之后,类上不需要添加任何东西。

            如果action作为目标对象切入事务,需要在<aop:config>元素里添加proxy-target-class="true"属性。原因是通知spring框架采用cglib技术生成具有事务管理功能的action类。

       -基于注解(配置简单,经常使用)--见demo4包

            在applicationcontext.xml中开启事务注解配置。(applicationcontext.xml中只需定义bean并追加以下元素)

?
1
2
3
4
<bean id="txmanager" class="...">
 <property name="sessionfactory">
 </property>
<tx:annotation-driven transaction-manager="txmanager"/>

            在目标组件类中使用@transactional,该标记可定义在类前或方法前。

四、示例(银行转账)

 

        --编程式

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
 * @description:转账案例的dao层接口
 *
 */
public interface accountdao {
 /**
 * @param out
 * :转出账号
 * @param money
 * :转账金额
 */
 public void outmoney(string out, double money);
 
 /**
 *
 * @param in
 * :转入账号
 * @param money
 * :转账金额
 */
 public void inmoney(string in, double money);
}
?
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
/**
 * @description:转账案例的dao层实现类
 */
public class accountdaoimpl extends jdbcdaosupport implements accountdao {
 /**
 * @param out
 * :转出账号
 * @param money
 * :转账金额
 */
 @override
 public void outmoney(string out, double money) {
 string sql = "update account set money = money-? where name = ?";
 this.getjdbctemplate().update(sql, money, out);
 }
 /**
 * @param in
 * :转入账号
 * @param money
 * :转账金额
 */
 @override
 public void inmoney(string in, double money) {
 string sql = "update account set money = money+? where name = ?";
 this.getjdbctemplate().update(sql, money, in);
 }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
/**
 * @description:转账案例的业务接口
 *
 */
public interface accountservice {
 /**
 * @param out :转出账号
 * @param in :转入账号
 * @param money :转账金额
 */
 public void transfer(string out,string in,double money);
}
?
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
45
46
/**
 * @description:转账案例的业务层实现类
 */
public class accountserviceimpl implements accountservice {
 // 注入转账的dao
 private accountdao accountdao;
 
 // 注入事务管理的模板
 private transactiontemplate transactiontemplate;
 
 /**
 * @param out
 * :转出账号
 * @param in
 * :转入账号
 * @param money
 * :转账金额
 */
 @override
 public void transfer(final string out, final string in, final double money) {
 
 // 未经事务控制的业务处理操作,如果过程中出异常,则导致前面的操作能完成,后面的不能,即转账成功但未收到转账款
 // accountdao.outmoney(out, money);
 // int i = 1/0;
 // accountdao.inmoney(in, money);
 
 transactiontemplate.execute(new transactioncallbackwithoutresult() {
 
 @override
 protected void dointransactionwithoutresult(
  transactionstatus transactionstatus) {
 accountdao.outmoney(out, money);
 // int i = 1 / 0;//事务控制,即出现异常,该段内代码都执行失效
 accountdao.inmoney(in, money);
 }
 });
 }
 
 public void setaccountdao(accountdao accountdao) {
 this.accountdao = accountdao;
 }
 
 public void settransactiontemplate(transactiontemplate transactiontemplate) {
 this.transactiontemplate = transactiontemplate;
 }
}

applicationcontext1.xml

?
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
<!-- 引入外部的属性文件 -->
 <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.username}" />
 <property name="password" value="${jdbc.password}" />
 </bean>
 
 <!-- 配置业务层类 -->
 <bean id="accountservice" class="com.zs.spring.demo1.accountserviceimpl">
 <property name="accountdao" ref="accountdao" />
 <!-- 注入事务管理的模板 -->
 <property name="transactiontemplate" ref="transactiontemplate" />
 </bean>
 
 <!-- 配置dao类(简化,会自动配置jdbctemplate) -->
 <bean id="accountdao" class="com.zs.spring.demo1.accountdaoimpl">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- 配置dao类(未简化) -->
 <!-- <bean id="jdbctemplate" class="org.springframework.jdbc.core.jdbctemplate">
 <property name="datasource" ref="datasource" />
 </bean>
 <bean id="accountdao" class="com.zs.spring.demo1.accountdaoimpl">
 <property name="jdbctemplate" ref="jdbctemplate" />
 </bean> -->
 
 <!-- ==================================1.编程式的事务管理=============================================== -->
 <!-- 配置事务管理器 -->
 <bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- 配置事务管理的模板:spring为了简化事务管理的代码而提供的类 -->
 <bean id="transactiontemplate" class="org.springframework.transaction.support.transactiontemplate">
 <property name="transactionmanager" ref="transactionmanager"/>
 </bean>

测试:

?
1
2
3
4
5
6
7
8
9
10
11
@runwith(springjunit4classrunner.class)
@contextconfiguration("classpath:applicationcontext1.xml")
public class transactiontest {
 @resource(name = "accountservice")
 private accountservice accountservice;
 
 @test
 public void demo1() {
 accountservice.transfer("aaa", "bbb", 200d);
 }
}

    --基于transactionproxyfactorybean的方式

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
public class accountserviceimpl implements accountservice {
 // 注入转账的dao
 private accountdao accountdao;
 
 /**
 * @param out
 *  :转出账号
 * @param in
 *  :转入账号
 * @param money
 *  :转账金额
 */
 @override
 public void transfer(string out, string in, double money) {
 accountdao.outmoney(out, money);
 // int i = 1/0;
 accountdao.inmoney(in, money);
 }
 
 public void setaccountdao(accountdao accountdao) {
 this.accountdao = accountdao;
 }
}

applicationcontext2.xml

?
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
45
46
47
48
49
50
51
<!-- 引入外部的属性文件 -->
 <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.username}" />
 <property name="password" value="${jdbc.password}" />
 </bean>
 
 <!-- 配置业务层类 -->
 <bean id="accountservice" class="com.zs.spring.demo2.accountserviceimpl">
 <property name="accountdao" ref="accountdao" />
 </bean>
 
 <!-- 配置dao类(简化,会自动配置jdbctemplate) -->
 <bean id="accountdao" class="com.zs.spring.demo2.accountdaoimpl">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- ==================================2.使用xml配置声明式的事务管理(原始方式)=============================================== -->
 
 <!-- 配置事务管理器 -->
 <bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- 配置业务层的代理 -->
 <bean id="accountserviceproxy" class="org.springframework.transaction.interceptor.transactionproxyfactorybean">
 <!-- 配置目标对象 -->
 <property name="target" ref="accountservice" />
 <!-- 注入事务管理器 -->
 <property name="transactionmanager" ref="transactionmanager"></property>
 <!-- 注入事务的属性 -->
 <property name="transactionattributes">
  <props>
  <!--
   prop的格式:
   * propagation :事务的传播行为
   * isotation :事务的隔离级别
   * readonly :只读
   * -exception :发生哪些异常回滚事务
   * +exception :发生哪些异常不回滚事务
   -->
  <prop key="transfer">propagation_required</prop>
  <!-- <prop key="transfer">propagation_required,readonly</prop> -->
  <!-- <prop key="transfer">propagation_required,+java.lang.arithmeticexception</prop> -->
  </props>
 </property>
 </bean>

测试:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
@runwith(springjunit4classrunner.class)
@contextconfiguration("classpath:applicationcontext2.xml")
public class transactiontest {
 /**
 * 一定要注入代理类:因为代理类进行增强的操作
 */
 // @resource(name="accountservice")
 @resource(name = "accountserviceproxy")
 private accountservice accountservice;
 
 @test
 public void demo1() {
 accountservice.transfer("aaa", "bbb", 200d);
 }
}

    --基于xml配置

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
public class accountserviceimpl implements accountservice {
 // 注入转账的dao
 private accountdao accountdao;
 
 /**
 * @param out
 *  :转出账号
 * @param in
 *  :转入账号
 * @param money
 *  :转账金额
 */
 @override
 public void transfer(string out, string in, double money) {
 accountdao.outmoney(out, money);
 // int i = 1/0;
 accountdao.inmoney(in, money);
 
 }
 
 public void setaccountdao(accountdao accountdao) {
 this.accountdao = accountdao;
 }
}

applicationcontext3.xml

?
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
45
46
47
48
49
50
<!-- 引入外部的属性文件 -->
 <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.username}" />
 <property name="password" value="${jdbc.password}" />
 </bean>
 
 <!-- 配置业务层类 -->
 <bean id="accountservice" class="com.zs.spring.demo3.accountserviceimpl">
 <property name="accountdao" ref="accountdao" />
 </bean>
 
 <!-- 配置dao类(简化,会自动配置jdbctemplate) -->
 <bean id="accountdao" class="com.zs.spring.demo3.accountdaoimpl">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- ==================================3.使用xml配置声明式的事务管理,基于tx/aop=============================================== -->
 
 <!-- 配置事务管理器 -->
 <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 :只读
  rollback-for:发生哪些异常回滚
  no-rollback-for :发生哪些异常不回滚
  timeout :过期信息
  -->
  <tx:method name="transfer" propagation="required"/>
 </tx:attributes>
 </tx:advice>
 
 <!-- 配置切面 -->
 <aop:config>
 <!-- 配置切入点 -->
 <aop:pointcut expression="execution(* com.zs.spring.demo3.accountservice+.*(..))" id="pointcut1"/>
 <!-- 配置切面 -->
 <aop:advisor advice-ref="txadvice" pointcut-ref="pointcut1"/>
 </aop:config>

测试:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
/**
 * @description:spring的声明式事务管理的方式二:基于aspectj的xml方式的配置
 */
@runwith(springjunit4classrunner.class)
@contextconfiguration("classpath:applicationcontext3.xml")
public class transactiontest {
 /**
 * 一定要注入代理类:因为代理类进行增强的操作
 */
 @resource(name = "accountservice")
 private accountservice accountservice;
 
 @test
 public void demo1() {
 accountservice.transfer("aaa", "bbb", 200d);
 }
}

    --基于注解

?
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
/**
 * @transactional中的的属性 propagation :事务的传播行为 isolation :事务的隔离级别 readonly :只读
 *   rollbackfor :发生哪些异常回滚 norollbackfor :发生哪些异常不回滚
 *   rollbackforclassname 根据异常类名回滚
 */
@transactional(propagation = propagation.required, isolation = isolation.default, readonly = false)
public class accountserviceimpl implements accountservice {
 // 注入转账的dao
 private accountdao accountdao;
 
 /**
 * @param out
 *  :转出账号
 * @param in
 *  :转入账号
 * @param money
 *  :转账金额
 */
 @override
 public void transfer(string out, string in, double money) {
 accountdao.outmoney(out, money);
 // int i = 1/0;
 accountdao.inmoney(in, money);
 }
 
 public void setaccountdao(accountdao accountdao) {
 this.accountdao = accountdao;
 }
}

applicationcontext4.xml

?
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
<!-- 引入外部的属性文件 -->
 <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.username}" />
 <property name="password" value="${jdbc.password}" />
 </bean>
 
 <!-- 配置业务层类 -->
 <bean id="accountservice" class="com.zs.spring.demo4.accountserviceimpl">
 <property name="accountdao" ref="accountdao" />
 </bean>
 
 <!-- 配置dao类(简化,会自动配置jdbctemplate) -->
 <bean id="accountdao" class="com.zs.spring.demo4.accountdaoimpl">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- ==================================4.使用注解配置声明式事务============================================ -->
 
 <!-- 配置事务管理器 -->
 <bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager">
 <property name="datasource" ref="datasource" />
 </bean>
 
 <!-- 开启注解事务 -->
 <tx:annotation-driven transaction-manager="transactionmanager"/>

测试:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
@runwith(springjunit4classrunner.class)
@contextconfiguration("classpath:applicationcontext4.xml")
public class transactiontest {
 /**
 * 一定要注入代理类:因为代理类进行增强的操作
 */
 @resource(name = "accountservice")
 private accountservice accountservice;
 
 @test
 public void demo1() {
 accountservice.transfer("aaa", "bbb", 200d);
 }
}

具体代码和数据库文件参考项目完整代码:

spring-transaction.rar

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。

原文链接:https://blog.csdn.net/daijin888888/article/details/51822257