前言
本文配套示例代码下载地址(完整可运行,含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); } } |
具体代码和数据库文件参考项目完整代码:
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:https://blog.csdn.net/daijin888888/article/details/51822257