声明式事务
Posted zhzcode
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了声明式事务相关的知识,希望对你有一定的参考价值。
【事务概述】
1) 在JavaEE企业级开发的应用领域,为了保证数据的完整性和一致性,必须引入数据库事务的概念,所以事务管理是企业级应用程序开发中必不可少的技术。
2) 事务就是一组由于逻辑上紧密关联而合并成一个整体(工作单元)的多个数据库操作,这些操作要么都执行,要么都不执行。
3) 事务的四个关键属性(ACID)
①原子性(atomicity):“原子”的本意是“不可再分”,事务的原子性表现为一个事务中涉及到的多个操作在逻辑上缺一不可。事务的原子性要求事务中的所有操作要么都执行,要么都不执行。
②一致性(consistency):“一致”指的是数据的一致,具体是指:所有数据都处于满足业务规则的一致性状态。一致性原则要求:一个事务中不管涉及到多少个操作,都必须保证事务执行之前数据是正确的,事务执行之后数据仍然是正确的。如果一个事务在执行的过程中,其中某一个或某几个操作失败了,则必须将其他所有操作撤销,将数据恢复到事务执行之前的状态,这就是回滚。
③隔离性(isolation):在应用程序实际运行过程中,事务往往是并发执行的,所以很有可能有许多事务同时处理相同的数据,因此每个事务都应该与其他事务隔离开来,防止数据损坏。隔离性原则要求多个事务在并发执行过程中不会互相干扰。
④持久性(durability):持久性原则要求事务执行完成后,对数据的修改永久的保存下来,不会因各种系统错误或其他意外情况而受到影响。通常情况下,事务对数据的修改应该被写入到持久化存储器中。
【Spring事务管理】
【编程式事务管理】
1) 使用原生的JDBC API进行事务管理
①获取数据库连接Connection对象
②取消事务的自动提交
③执行操作
④正常完成操作时手动提交事务
⑤执行失败时回滚事务
⑥关闭相关资源
2) 评价
使用原生的JDBC API实现事务管理是所有事务管理方式的基石,同时也是最典型的编程式事务管理。编程式事务管理需要将事务管理代码嵌入到业务方法中来控制事务 的提交和回滚。在使用编程的方式管理事务时,必须在每个事务操作中包含额外的事务 管理代码。相对于核心业务而言,事务管理的代码显然属于非核心业务,如果多个模块 都使用同样模式的代码进行事务管理,显然会造成较大程度的代码冗余。
【声明式事务管理】
大多数情况下声明式事务比编程式事务管理更好:它将事务管理代码从业务方法中分离出来,以声明的方式来实现事务管理。
事务管理代码的固定模式作为一种横切关注点,可以通过AOP方法模块化,进而借助Spring AOP框架实现声明式事务管理。
Spring在不同的事务管理API之上定义了一个抽象层,通过配置的方式使其生效,从而让应用程序开发人员不必了解事务管理API的底层实现细节,就可以使用Spring的事务管理机制。
Spring既支持编程式事务管理,也支持声明式的事务管理。
【事务管理器的主要实现】
1) DataSourceTransactionManager:在应用程序中只需要处理一个数据源,而且通过JDBC存取。
2) JtaTransactionManager:在JavaEE应用服务器上用JTA(Java Transaction API)进行事务管理
3) HibernateTransactionManager:用Hibernate框架存取数据库
【测试数据准备】
1.需求
2.数据库表
CREATE TABLE book (
isbn VARCHAR (50) PRIMARY KEY,
book_name VARCHAR (100),
price INT
) ;
CREATE TABLE book_stock (
isbn VARCHAR (50) PRIMARY KEY,
stock INT,
CHECK (stock > 0)
) ;
CREATE TABLE account (
username VARCHAR (50) PRIMARY KEY,
balance INT,
CHECK (balance > 0)
) ;
INSERT INTO account (`username`,`balance`) VALUES (‘Tom‘,100000);
INSERT INTO account (`username`,`balance`) VALUES (‘Jerry‘,150000);
INSERT INTO book (`isbn`,`book_name`,`price`) VALUES (‘ISBN-001‘,‘book01‘,100);
INSERT INTO book (`isbn`,`book_name`,`price`) VALUES (‘ISBN-002‘,‘book02‘,200);
INSERT INTO book (`isbn`,`book_name`,`price`) VALUES (‘ISBN-003‘,‘book03‘,300);
INSERT INTO book (`isbn`,`book_name`,`price`) VALUES (‘ISBN-004‘,‘book04‘,400);
INSERT INTO book (`isbn`,`book_name`,`price`) VALUES (‘ISBN-005‘,‘book05‘,500);
INSERT INTO book_stock (`isbn`,`stock`) VALUES (‘ISBN-001‘,1000);
INSERT INTO book_stock (`isbn`,`stock`) VALUES (‘ISBN-002‘,2000);
INSERT INTO book_stock (`isbn`,`stock`) VALUES (‘ISBN-003‘,3000);
INSERT INTO book_stock (`isbn`,`stock`) VALUES (‘ISBN-004‘,4000);
INSERT INTO book_stock (`isbn`,`stock`) VALUES (‘ISBN-005‘,5000);
【初步实现】
1.配置文件
1 <!-- 配置事务管理器 --> 2 <bean id="transactionManager" 3 class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> 4 <property name="dataSource" ref="dataSource"/> 5 </bean> 6 7 <!-- 启用事务注解 --> 8 <tx:annotation-driven transaction-manager="transactionManager"/>
2. 在需要进行事务控制的方法上加注解 @Transactional
以上是关于声明式事务的主要内容,如果未能解决你的问题,请参考以下文章