编程开源技术交流,分享技术与知识

网站首页 > 开源技术 正文

Spring基础笔记(三)- 事务控制(spring事务控制实现)

wxchong 2024-07-23 21:22:00 开源技术 29 ℃ 0 评论

1. 事务

1.1 4个基本特性(ACID)

4个基本属性:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability):

  1. 原子性:事务内容要么全部执行成功,要么全部执行失败,成功则写入数据库,失败则对数据库不产生影响;
  2. 隔离性:用户并发访问时会同时存在多个事务,多个并发事务之间相互隔离;
  3. 一致性:事务执行前和执行后状态保持一致,比如A向B转账,A扣钱后B必定收到等额入账,且总数不变;
  4. 持久性:事务完成后,事务对数据库的所有更新执行都永久性保存到数据库,不能再被回滚;

1.2 事务并发问题

多个事务并发时可能出现以下并发问题:

  1. 脏读:事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据;
  2. 不可重复读:事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据的结果不一致;
  3. 幻读:系统管理员A将数据库中所有学生的成绩从具体分数改为ABCDE等级,但是系统管理员B就在这个时候插入了一条具体分数的记录,当系统管理员A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。

小结:不可重复读的和幻读很容易混淆,不可重复读侧重于修改,幻读侧重于新增或删除。解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表

1.3 事务隔离级别

为了处理事务并发问题,可以为事务设置不同的隔离级别。

MySQL数据库支持4种隔离级别:

  1. Read Uncommitted(读未提交):最低级别,任何情况都可能发生;
  2. Read Committed(读已提交):可避免脏读的发生;
  3. Repeatable Read(可重复读):可避免脏读、不可重复读的发生;
  4. Serializable(串行化):最高级别,避免脏读、不可重复读、幻读的发生;

需要注意一下几点:

  1. 四种隔离级别最高的是Seralizable级别,最低的是Read Uncommitted级别,级别越高执行效率就越低;
  2. 隔离级别的设置只对当前链接有效,对JDBC操作数据库来说,一个Connection对象相当于一个链接;
  3. Mysql的默认隔离级别是:可重复读(Repeatable Read);
  4. Oracle只支持串行化(Seralizable)级别和读已提交(Read committed)级别,默认是读已提交(Read Committed)级别。
  5. 采用最高事务隔离级别Serializable(串行化)时事务顺序执行,能避免并发问题,但性能很低,很少使用。

验证事务隔离级别:Read-Uncommitted(读未提交),其它级别操作雷同,故不再演示;

打开2个MySQL客户端,先修改隔离级别为Read Uncommitted,然后依次按照

第一步:客户端A——修改当前连接的隔离级别,查询表t_user的初始状态

 mysql> setsessiontransactionisolationlevelreaduncommitted;
 QueryOK, 0rows affected (0.00sec)
 ?
 mysql> select* fromt_user;
 +----+-----------+-----------+
 | id | user_name | nick_name |
 +----+-----------+-----------+
 | 1| qiaofeng | qf |
 +----+-----------+-----------+
 1rowinset(0.00sec)

第二步:客户端B——开启事务,插入一条数据但不提交

 mysql> setsessiontransactionisolationlevelreaduncommitted;
 QueryOK, 0rows affected (0.00sec)
 ?
 mysql> starttransaction;
 QueryOK, 0rows affected (0.00sec)
 ?
 mysql> insertintot_user(user_name,nick_name)value('xuzhu','xz');
 QueryOK, 1rowaffected (0.00sec)
 ?

第三步:客户端A——当前隔离级别下,A可以查询到B未提交的数据

 mysql> select* fromt_user;
 +----+-----------+-----------+
 | id | user_name | nick_name |
 +----+-----------+-----------+
 | 1| qiaofeng | qf |
 | 2| xuzhu | xz |
 +----+-----------+-----------+
 2rows inset(0.00sec)

第四步:客户端B——回滚事务

 mysql> rollback;
 QueryOK, 0rows affected (0.00sec)

第五步:客户端A——B事务回滚后,再次查询表的数据

 mysql> select* fromt_user;
 +----+-----------+-----------+
 | id | user_name | nick_name |
 +----+-----------+-----------+
 | 1| qiaofeng | qf |
 +----+-----------+-----------+
 1rowinset(0.00sec)

2. Spring的事务管理

2.1 概述

  • 作为企业级应用程序框架,Spring 在不同的事务管理 API 之上定义了一个抽象层,而应用程序开发人员不必了解底层的事务管理 API, 就可以使用 Spring 的事务管理机制;有了这些事务机制,事务管理器代码就能独立于特定的事务技术了;
  • Spring 既支持编程式事务管理,也支持声明式的事务管理;
  • 编程式事务管理:将事务管理代码嵌入到业务方法中来控制事务的提交和回滚;在编程式管理事务时,必须在每个事务操作中包含额外的事务管理代码;
  • 声明式事务管理:大多数情况下比编程式事务管理更好用;它将事务管理代码从业务方法中分离出来, 以声明的方式来实现事务管理;事务管理作为一种横切关注点,可以通过 AOP 方法模块化,Spring 通过 Spring AOP 框架支持声明式事务管理;
  • Spring 的核心事务管理抽象是org.springframework.transaction.PlatformTransactionManager, 它为事务管理封装了一组独立于技术的方法;无论使用Spring的哪种事务管理策略(编程式或声明式),事务管理器都是必须的。
  • 事务管理器以普通的 Bean 形式声明在Spring IOC 容器中;

2.2 常见的事务管理器

  1. org.springframework.jdbc.datasource.DataSourceTransactionManager : 应用程序只需要处理一个数据源,并且通过JDBC存取;
  2. org.springframework.transaction.jta.JtaTransactionManager : 在JavaEE应用服务器上用JTA(Java Transaction API)进行事务管理;
  3. org.springframework.orm.hibernate3.HibernateTransactionManager :使用Hibernate框架存取数据库并进行事务管理;

2.3 使用@Transactional实现的声明式事务管理:

  • 除了在带有切入点、通知和增强器的 Bean 配置文件中声明事务外, Spring 还允许简单地用 @Transactional 注解来标注事务方法;
  • 为了将方法定义为支持事务处理的,可以为方法添加 @Transactional 注解,根据 Spring AOP 基于代理机制,只能标注公有方法;
  • 可以在方法或者类级别上添加 @Transactional 注解,当把这个注解应用到类上时,这个类中的所有公共方法都会被定义成支持事务处理的;
  • 在 Bean 配置文件中只需要启用 tx:annotation-driven 元素,并为之指定事务管理器就可以了;
  • 如果事务处理器的名称是 transactionManager,就可以在tx:annotation-driven 元素中省略 transaction-manager 属性,这个元素会自动检测该名称的事务处理器。

2.4 使用事务通知实现的声明式事务管理:

  • 事务管理是一种横切关注点;
  • 为了在 Spring 2.x 中启用声明式事务管理,可以通过 tx Schema 中定义的 tx:advice 元素声明事务通知,为此必须事先将这个 Schema 定义添加到 <beans> 根元素中去;
  • 声明了事务通知后, 就需要将它与切入点关联起来;由于事务通知是在 aop:config 元素外部声明的,所以它无法直接与切入点产生关联,所以必须在 aop:config 元素中声明一个增强器通知与切入点关联起来;
  • 由于 Spring AOP 是基于代理的方法,所以只能增强公共方法;因此,只有公有方法才能通过 Spring AOP 进行事务管理。

3. 基于注解实现的事务控制

3.1 小需求

关于事务的小需求:用户购买指定图书,图书有一定库存,用户有一定的金钱,每次只能购买一本图书,购买需要花费对应书价的金钱,购买方法使用事务管理,过程分为三步:

  1. 根据书号ISBN号查询图书价格;
  2. 扣减图书库存,每次只能扣减一本,库存不足时抛出异常;
  3. 扣减用户账户余额,扣减数目与图书价格相同,余额不足时购买失败并抛出异常。

3.2 关键实现代码

第一步:初始化表结构

 insertintot_book(isbn,book_name,price)values('IB1234','Java编程思想',50);
 insertintot_book(isbn,book_name,price)values('IB1235','C++高级编程',100);
 ?
 insertintot_book_stock values('IB1234', 20),('IB1235',20);
 insertintot_account values('qiaofeng',50),('duanyu',150);

第二步:引入指定包

 <!-- https://mvnrepository.com/artifact/org.springframework/spring-tx -->
 <dependency>
 <groupId>org.springframework</groupId>
 <artifactId>spring-tx</artifactId>
 <version>5.1.3.RELEASE</version>
 </dependency>
 <!-- https://mvnrepository.com/artifact/org.springframework/spring-jdbc -->
 <dependency>
 <groupId>org.springframework</groupId>
 <artifactId>spring-jdbc</artifactId>
 <version>${spring.version}</version>
 </dependency>
 <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
 <dependency>
 <groupId>mysql</groupId>
 <artifactId>mysql-connector-java</artifactId>
 <version>6.0.6</version>
 </dependency>
 <!-- https://mvnrepository.com/artifact/com.mchange/c3p0 -->
 <dependency>
 <groupId>com.mchange</groupId>
 <artifactId>c3p0</artifactId>
 <version>0.9.5.1</version>
 </dependency>

第三步:XML配置关键信息

 <?xmlversion="1.0" encoding="UTF-8"?>
 <beansxmlns="http://www.springframework.org/schema/beans"
 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xmlns:context="http://www.springframework.org/schema/context"
 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/tx 
 http://www.springframework.org/schema/tx/spring-tx.xsd">
 <!-- 开启注解扫描,指定扫描的包范围 -->
 <context:component-scanbase-package="com.tengol.demo.spring.tx"/>
 
 <!-- 加载属性文件 -->
 <context:property-placeholderlocation="classpath:db-tx.properties"/>
 <!-- 配置连接池 -->
 <beanid="dataSource"class="com.mchange.v2.c3p0.ComboPooledDataSource">
 <propertyname="driverClass"value="${jdbc.driverClass}"/>
 <propertyname="jdbcUrl"value="${jdbc.url}"/>
 <propertyname="user"value="${jdbc.user}"/>
 <propertyname="password"value="${jdbc.password}"/>
 </bean>
 <!-- 配置JdbcTemplate模板 -->
 <beanid="jdbcTemplate"class="org.springframework.jdbc.core.JdbcTemplate">
 <propertyname="dataSource"ref="dataSource"/>
 </bean>
 <!-- 事务管理器,此处使用数据源的事务管理器 -->
 <beanid="transactionManager"
 class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
 <propertyname="dataSource"ref="dataSource"/>
 </bean>
 <!-- 开启事务注解 -->
 <tx:annotation-driven/>
 
 </beans>

第四步:数据库信息的属性文件

 jdbc.driverClass=com.mysql.jdbc.Driver
 jdbc.url=jdbc:mysql://localhost:3306/jwms_auth?useSSL=true
 jdbc.user=root
 jdbc.password=Jwms@2018

第五步:关键代码-业务层

 publicinterfaceBookShopService{
 /**
 * 模拟用户购买图书
 */
 publicvoidpurchase(Stringusername, Stringisbn);
 }
 ?
 @Service
 publicclassBookShopServiceImplimplementsBookShopService{
 /**
 * 图书销售的持久化操作类
 */
 privateBookShopDaobookShopDao;
 /**
 * Spring推荐使用构造器或Setter方式进行强依赖注入,不推荐属性注入
 */
 @Autowired
 publicBookShopServiceImpl(BookShopDaobookShopDao) {
 this.bookShopDao=bookShopDao;
 }
 ?
 @Transactional
 @Override
 publicvoidpurchase(Stringusername, Stringisbn) {
 //1.获取图书的单价
 IntegerbookPrice=bookShopDao.findBookPriceByIsbn(isbn);
 System.out.println("图书"+isbn+"的单价为:"+bookPrice);
 //2.更新图书的库存
 bookShopDao.updateBookStock(isbn);
 System.out.println("图书"+isbn+"的库存成功被扣减1个");
 //3.更新用户的账户
 bookShopDao.updateUserAccount(username, bookPrice);
 System.out.println("用户"+username+"花费掉"+bookPrice+"元购买了图书"+isbn);
 }
 }

第六步:关键代码-持久层

 publicinterfaceBookShopDao{
 /**
 * 根据书号获取图书的销售单价
 */
 publicIntegerfindBookPriceByIsbn(Stringisbn);
 ?
 /**
 * 更新图书库存,使库存减一个(stock=stock-1)
 */
 publicvoidupdateBookStock(Stringisbn);
 ?
 /**
 * 更新用户账户,使用户当前余额减去price(balance=balance-price)
 */
 publicvoidupdateUserAccount(Stringusername, Integerprice);
 }
 ?
 @Repository
 publicclassBookShopDaoImplimplementsBookShopDao{
 ?
 /**
 * Spring的JDBC操作模板
 */
 privateJdbcTemplatejdbcTemplate;
 ?
 /**
 * 用于依赖注入的构造器
 */
 @Autowired
 publicBookShopDaoImpl(JdbcTemplatejdbcTemplate) {
 this.jdbcTemplate=jdbcTemplate;
 }
 ?
 /**
 * 根据书号获取图书的销售单价
 */
 @Override
 publicIntegerfindBookPriceByIsbn(Stringisbn) {
 Stringsql="select price from t_book where isbn = ?";
 IntegerbookPrice=jdbcTemplate.queryForObject(sql, Integer.class, isbn);
 returnbookPrice;
 }
 ?
 /**
 * 更新图书库存,使库存减一个(stock=stock-1)
 */
 @Override
 publicvoidupdateBookStock(Stringisbn) {
 //检查书的库存是否足够, 若不够, 则抛出异常
 StringcheckSql="select stock from t_book_stock where isbn=?";
 IntegercurrentStock=jdbcTemplate.queryForObject(checkSql, Integer.class, isbn);
 if(currentStock==null||currentStock<=0){
 thrownewBookStockException("图书"+isbn+"库存不足!");
 }
 Stringsql="update t_book_stock set stock=stock-1 where isbn=?";
 jdbcTemplate.update(sql, isbn);
 }
 ?
 /**
 * 更新用户账户,使用户当前余额减去price(balance=balance-price)
 */
 @Override
 publicvoidupdateUserAccount(Stringusername, Integerprice) {
 //检查用户余额是否充裕
 StringcheckSql="select balance from t_account where username=?";
 Integerbalance=jdbcTemplate.queryForObject(checkSql, Integer.class, username);
 if(balance==null||balance<price){
 thrownewUserAccountException("用户"+username+"的账户余额不足!");
 }
 Stringsql="update t_account set balance=balance-? where username=?";
 jdbcTemplate.update(sql, price, username);
 }
 }

第七步:关键代码-测试用例

 publicclassTransactionTest{
 ?
 privateApplicationContextcontext;
 privateBookShopDaobookShopDao;
 privateBookShopServicebookShopService;
 privateCashierServicecashierService;
 ?
 {
 context=newClassPathXmlApplicationContext("spring-tx.xml");
 bookShopDao=context.getBean(BookShopDao.class);
 bookShopService=context.getBean(BookShopService.class);
 cashierService=context.getBean(CashierService.class);
 }
 
 /**
 * 测试事务控制是否生效
 * (1) 假如库存不足,用户扣款是否成功 (答案:否,还未执行到)
 * (2) 假如库存充裕,用户账户余额不足,是否库存扣减成功 (答案:否,错误回滚)
 * (3) 假如库存充裕,用于账户余额充裕,是否可以购买成功 (答案:是,能够正确执行)
 * (4) 假如去掉事务控制标签@Transactional,上述事务控制是否生效 (答案:否,只有部分成功)
 */
 @Test
 publicvoidtestPurchase(){
 bookShopService.purchase("qiaofeng","IB1234");
 }
 ?
 @Test
 publicvoidtestFindBookPrice() {
 System.out.println(bookShopDao.findBookPriceByIsbn("IB1234"));
 }
 ?
 @Test
 publicvoidtestUpdateBookStock() {
 Stringisbn="IB1235";
 bookShopDao.updateBookStock(isbn);
 System.out.println("图书"+isbn+"的库存更新成功");
 }
 ?
 @Test
 publicvoidtestUpdateUserAccount() {
 StringuserName="qiaofeng";
 Integerprice=50;
 bookShopDao.updateUserAccount(userName, price);
 System.out.println("用户"+userName+"买书花费了"+price);
 }
 }
 ?

3.3 细节-事务传播机制

3.3.1 概述

当事务方法被另一个事务方法调用时,必须指定事务应该如何传播,例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。

事务的传播行为可以由传播属性指定,Spring 定义了 7 种类传播行为(如下图所示),在实际工作中,最常使用的事务传播行为是:REQUIRED 和 REQUIRED_NEW。

3.3.2 关键代码

为了方便测试,增加一个客户结账操作(CashierService),用于结算用户购买的多本图书,当用户余额不足时可指定用户的购买行为:只能买N本,或者,一本都买不了。

需要设置:

  1. 用户qiaofeng账户余额为50元,图书IB1234售价为50元,库存为20;图书IB1235售价为100元,库存为20;
  2. 用户qiaofeng账户余额只能购买一本图书IB1234,无法购买图书IB1235;

假如用户要购买2本不同图书,在此场景下验证事务传播机制:

增加结账操作(CashierService)

 publicinterfaceCashierService{
 /**
 * 客户结账操作:用户购买多本图书
 */
 publicvoidcheckout(Stringusername, List<String>isbnList);
 }
 ?
 @Service
 publicclassCashierServiceImplimplementsCashierService{
 privateBookShopServicebookShopService;
 ?
 @Autowired
 publicCashierServiceImpl(BookShopServicebookShopService) {
 this.bookShopService=bookShopService;
 }
 ?
 /**
 * 客户结账操作:用户购买多本图书
 */
 @Transactional
 @Override
 publicvoidcheckout(Stringusername, List<String>isbnList) {
 for(Stringisbn: isbnList) {
 //调用BookShopService的purchase方法,购买一本图书
 bookShopService.purchase(username, isbn);
 }
 }
 }
 ?

3.3.3 验证REQUIRED行为

REQUIRED行为是默认的事务传播行为,也可以设置属性propagation = Propagation.REQUIRED;

表示purchase()方法使用checkout()方法的事务处理该方法的操作,故由于用户账户余额不足,导致checkout方法的2本图书均无法购买成功,也就是用户一本都没买成。

在checkout()方法的开始和终止边界内只有一个事务,这个事务只在checkout()方法结束的时候被提交,结果用户一本书都买不了,过程如下图所示:

代码实现如下:

 @Transactional
 @Override
 publicvoidcheckout(Stringusername, List<String>isbnList) {
 for(Stringisbn: isbnList) {
 bookShopService.purchase(username, isbn);
 }
 }
 ?
 @Transactional(propagation=Propagation.REQUIRED)
 @Override
 publicvoidpurchase(Stringusername, Stringisbn) {
 //1.获取图书的单价
 IntegerbookPrice=bookShopDao.findBookPriceByIsbn(isbn);
 //2.更新图书的库存
 bookShopDao.updateBookStock(isbn);
 //3.更新用户的账户
 bookShopDao.updateUserAccount(username, bookPrice);
 }

3.3.4 验证REQUIRED_NEW模式

另一种常见的传播行为是 REQUIRES_NEW,它表示该方法必须启动一个新事务,并在自己的事务内运行;如果有事务在运行,就应该先挂起它。

代码实现如下:

在方法purchase()中设置propagation = Propagation.REQUIRES_NEW,则调用此方法时开启本方法的新事务;

 @Transactional
 @Override
 publicvoidcheckout(Stringusername, List<String>isbnList) {
 for(Stringisbn: isbnList) {
 bookShopService.purchase(username, isbn);
 }
 }
 ?
 @Transactional(propagation=Propagation.REQUIRES_NEW)
 @Override
 publicvoidpurchase(Stringusername, Stringisbn) {
 //1.获取图书的单价
 IntegerbookPrice=bookShopDao.findBookPriceByIsbn(isbn);
 //2.更新图书的库存
 bookShopDao.updateBookStock(isbn);
 //3.更新用户的账户
 bookShopDao.updateUserAccount(username, bookPrice);
 }

3.4 细节-隔离级别

事务的4种隔离级别(级别右低变高):

  1. Read Uncommitted(读未提交):最低级别,任何情况都可能发生;
  2. Read Committed(读已提交):可避免脏读的发生;
  3. Repeatable Read(可重复读):可避免脏读、不可重复读的发生;
  4. Serializable(串行化):最高级别,避免脏读、不可重复读、幻读的发生;

关于事务的隔离级别,需要注意的是:

  1. 从理论上来说,事务应该彼此完全隔离,以避免并发事务所导致的问题,然而,那样会对性能产生极大的影响, 因为事务必须按顺序运行;
  2. 在实际开发中,为了提升性能,事务会以较低的隔离级别运行;
  3. MySQL数据库支持4种事务隔离级别,Oracle数据库只支持其中2种:READ_COMMITED , SERIALIZABLE;
  4. 事务的隔离级别要得到底层数据库引擎的支持,而不是应用程序或者框架的支持。

事务的隔离级别可以通过隔离事务属性指定,指定属性isolation = Isolation.READ_COMMITTED。

 @Transactional(isolation=Isolation.READ_COMMITTED)
 @Override
 publicvoidpurchase(Stringusername, Stringisbn) {
 //1.获取图书的单价
 IntegerbookPrice=bookShopDao.findBookPriceByIsbn(isbn);
 //2.更新图书的库存
 bookShopDao.updateBookStock(isbn);
 //3.更新用户的账户
 bookShopDao.updateUserAccount(username, bookPrice);
 }

3.5 细节-设置回滚事务属性

默认情况下只有未检查异常(RuntimeException和Error类型的异常)会导致事务回滚,而受检查异常不会;

事务的回滚规则可以通过 @Transactional 注解的 rollbackFor 和 noRollbackFor 属性来定义,这两个属性被声明为 Class[] 类型的,因此可以为这两个属性指定多个异常类。

  • rollbackFor: 遇到时必须进行回滚
  • noRollbackFor:一组异常类,遇到时必须不回滚
 @Transactional(propagation=Propagation.REQUIRES_NEW,
 isolation=Isolation.READ_COMMITTED,
 rollbackFor={IOException.class, SQLException.class},
 noRollbackFor=ArithmeticException.class)
 @Override
 publicvoidpurchase(Stringusername, Stringisbn) {
 //1.获取图书的单价
 IntegerbookPrice=bookShopDao.findBookPriceByIsbn(isbn);
 //2.更新图书的库存
 bookShopDao.updateBookStock(isbn);
 //3.更新用户的账户
 bookShopDao.updateUserAccount(username, bookPrice);
 }

3.6 细节-超时和只读属性

由于事务可以在行和表上获得锁,因此长事务会占用资源,并对整体性能产生影响;

如果一个事物只读取数据但不做修改,数据库引擎可以对这个事务进行优化;

  • 超时事务属性:事务在强制回滚之前可以保持多久,这样可以防止长期运行的事务占用资源;
  • 只读事务属性:表示这个事务只读取数据但不更新数据,这样可以帮助数据库引擎优化事务。

超时和只读属性可以在@Transactional注解中定义,超时属性以秒为单位来计算。

 @Transactional(propagation=Propagation.REQUIRES_NEW,
 isolation=Isolation.READ_COMMITTED,
 rollbackFor={IOException.class, SQLException.class},
 noRollbackFor=ArithmeticException.class,
 readOnly=false,
 timeout=5)
 @Override
 publicvoidpurchase(Stringusername, Stringisbn) {
 //1.获取图书的单价
 IntegerbookPrice=bookShopDao.findBookPriceByIsbn(isbn);
 //2.更新图书的库存
 bookShopDao.updateBookStock(isbn);
 //3.更新用户的账户
 bookShopDao.updateUserAccount(username, bookPrice);
 }

4. 基于XML配置实现的事务控制

4.1 事务配置

 <!-- 定义持久层 -->
 <beanid="bookShopDao"class="com.tengol.demo.spring.tx.dao.BookShopDaoImpl">
 <constructor-argname="jdbcTemplate"ref="jdbcTemplate"/>
 </bean>
 <!-- 定义业务层 -->
 <beanid="bookShopService"class="com.tengol.demo.spring.tx.service.BookShopServiceImpl">
 <constructor-argname="bookShopDao"ref="bookShopDao"/>
 </bean>
 <!-- 事务配置第一步:声明事务管理器 -->
 <beanid="transactionManager" 
 class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
 <propertyname="dataSource"ref="dataSource"/>
 </bean>
 <!-- 事务配置第二步:声明事务通知 -->
 <tx:adviceid="bookShopTxAdvice"transaction-manager="transactionManager"/>
 <!-- 事务配置第三步:声明事务管理的范围 -->
 <aop:config>
 <aop:pointcutid="bookShopPoint"expression="execution(* *.BookShopService.*(..))"/>
 <aop:advisoradvice-ref="bookShopTxAdvice"pointcut-ref="bookShopPoint"/>
 </aop:config>

4.2 细节-事务传播机制

 <!-- 事务配置第二步:声明事务通知 -->
 <tx:adviceid="bookShopTxAdvice"transaction-manager="transactionManager">
 <tx:attributes>
 <tx:methodname="purchase"propagation="REQUIRES_NEW"/>
 </tx:attributes>
 </tx:advice>

4.3 细节-隔离级别

 <!-- 事务配置第二步:声明事务通知 -->
 <tx:adviceid="bookShopTxAdvice"transaction-manager="transactionManager">
 <tx:attributes>
 <tx:methodname="purchase"propagation="REQUIRES_NEW"
 isolation="READ_COMMITTED"
 roo/>
 </tx:attributes>
 </tx:advice>

4.4 细节-设置回滚事务属性

 <!-- 事务配置第二步:声明事务通知 -->
 <tx:adviceid="bookShopTxAdvice"transaction-manager="transactionManager">
 <tx:attributes>
 <tx:methodname="purchase"propagation="REQUIRES_NEW"isolation="READ_COMMITTED"
 rollback-for="java.io.IOException,java.sql.SQLException"
 no-rollback-for="java.lang.ArithmeticException"/>
 </tx:attributes>
 </tx:advice>

4.5 细节-超时和只读属性

 <!-- 事务配置第二步:声明事务通知 -->
 <tx:adviceid="bookShopTxAdvice"transaction-manager="transactionManager">
 <tx:attributes>
 <tx:methodname="purchase"propagation="REQUIRES_NEW"isolation="READ_COMMITTED"
 rollback-for="java.io.IOException,java.sql.SQLException"
 no-rollback-for="java.lang.ArithmeticException"
 read-only="false"
 timeout="5"/>
 </tx:attributes>
 </tx:advice>

5. Spring Jdbc Template

为了使 JDBC 更加易于使用,Spring 在 JDBC API 上定义了一个抽象层,以此建立一个JDBC存取框架;

JdbcTemplate 类被设计成为线程安全的,故可以在IOC容器中声明它的单个实例,并注入给所有的DAO实例;

常用操作如下:

5.1 HelloWorld

第一步:引入jar包

 <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
 <dependency>
 <groupId>mysql</groupId>
 <artifactId>mysql-connector-java</artifactId>
 <version>6.0.6</version>
 </dependency>
 <!-- https://mvnrepository.com/artifact/org.springframework/spring-jdbc -->
 <dependency>
 <groupId>org.springframework</groupId>
 <artifactId>spring-jdbc</artifactId>
 <version>${spring.version}</version>
 </dependency>
 <!-- https://mvnrepository.com/artifact/com.mchange/c3p0 -->
 <dependency>
 <groupId>com.mchange</groupId>
 <artifactId>c3p0</artifactId>
 <version>0.9.5.1</version>
 </dependency>

第二步:XML配置及属性文件

 <?xmlversion="1.0" encoding="UTF-8"?>
 <beansxmlns="http://www.springframework.org/schema/beans"
 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xmlns:context="http://www.springframework.org/schema/context"
 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">
 <!-- 加载属性文件 -->
 <context:property-placeholderlocation="classpath:db.properties"/>
 <!-- 配置连接池 -->
 <beanid="dataSource"class="com.mchange.v2.c3p0.ComboPooledDataSource">
 <propertyname="driverClass"value="${jdbc.driverClass}"/>
 <propertyname="jdbcUrl"value="${jdbc.url}"/>
 <propertyname="user"value="${jdbc.user}"/>
 <propertyname="password"value="${jdbc.password}"/>
 </bean>
 <!-- 配置JdbcTemplate模板 -->
 <beanid="jdbcTemplate"class="org.springframework.jdbc.core.JdbcTemplate">
 <propertyname="dataSource"ref="dataSource"/>
 </bean>
 </beans>

第三步:编写测试用例

 publicclassJdbcTemplateTest{
 privateJdbcTemplatejdbcTemplate;
 ?
 @Before
 publicvoidinit(){
 ApplicationContextcontext=
 newClassPathXmlApplicationContext("spring-jdbc.xml");
 jdbcTemplate=context.getBean(JdbcTemplate.class);
 }
 ?
 @Test
 publicvoidtestUpdate(){
 Stringsql="insert into t_user(user_name,nick_name) value(?,?);";
 intupdateRecords=jdbcTemplate.update(sql, "duanyu", "dy");
 System.out.println("insert "+updateRecords+" records");
 }
 }

5.2 执行一条更新语句

方法:org.springframework.jdbc.core.JdbcTemplate#update(java.lang.String, java.lang.Object...)

 @Test
 publicvoidtestUpdate(){
 Stringsql="insert into t_user(user_name,nick_name) value(?,?);";
 intupdateRecords=jdbcTemplate.update(sql, "duanyu", "dy");
 System.out.println("insert "+updateRecords+" records");
 }

5.3 执行批量更新语句

 /**
 * 执行批量更新: 批量的 INSERT, UPDATE, DELETE
 * 最后一个参数是 Object[] 的 List 类型: 因为修改一条记录需要一个 Object 的数组, 
 * 那么多条不就需要多个 Object 的数组吗
 */
 @Test
 publicvoid testBatchUpdate(){
 Stringsql="insert into t_user(user_name,nick_name)value(?,?)";
 List<Object[]>args=newArrayList<Object[]>();
 args.add(newObject[]{"qiaofeng","qf"});
 args.add(newObject[]{"xuzhu","xz"});
 args.add(newObject[]{"duanyu","dy"});
 jdbcTemplate.batchUpdate(sql,args);
 }

5.4 查询一条结果

 /**
 * 从数据库中获取一条记录, 实际得到对应的一个对象
 * 注意不是调用 queryForObject(String sql, Class<User> requiredType, Object... args) 
 * 而需要调用 queryForObject(String sql, RowMapper<User> rowMapper, Object... args)
 * 1. 其中的 RowMapper 指定如何去映射结果集的行, 常用的实现类为 BeanPropertyRowMapper
 * 2. 使用 SQL 中列的别名完成列名和类的属性名的映射. 例如 user_name userName
 * 3. 不支持级联属性. JdbcTemplate 到底是一个 JDBC 的小工具, 而不是 ORM 框架
 */
 @Test
 publicvoidtestQueryForObject(){
 Stringsql="select user_name userName, nick_name nickName from t_user where id=?";
 RowMapper<User>userRowMapper=newBeanPropertyRowMapper<User>(User.class);
 Useruser=jdbcTemplate.queryForObject(sql, userRowMapper, 5);
 System.out.println(newGson().toJson(user));
 }

5.5 查询多条结果

 /**
 * 查到实体类的集合
 * 注意调用的不是 queryForList 方法
 */
 @Test
 publicvoidtestQueryForList(){
 Stringsql="select user_name userName, nick_name nickName from t_user where id > ?";
 RowMapper<User>userRowMapper=newBeanPropertyRowMapper<User>(User.class);
 List<User>userList=jdbcTemplate.query(sql, userRowMapper, 5);
 System.out.println(newGson().toJson(userList));
 }

5.6 获取单个列的值

 /**
 * 获取单个列的值, 或做统计查询
 * 使用 queryForObject(String sql, Class<Long> requiredType) 
 */
 @Test
 publicvoidtestQueryOneColumn(){
 Stringsql="select count(id) from t_user";
 Integercount=jdbcTemplate.queryForObject(sql, int.class);
 System.out.println(count);
 }

5.7 执行具名参数语句

 /**
 * 可以为参数起名字. 
 * 1. 好处: 若有多个参数, 则不用再去对应位置, 直接对应参数名, 便于维护
 * 2. 缺点: 较为麻烦. 
 */
 @Test
 publicvoidtestNamedParameter(){
 Stringsql="insert into t_user(user_name,nick_name)value(:uname,:nname)";
 Map<String,Object>args=newHashMap<String, Object>();
 args.put("uname","xiaofeng");
 args.put("nname","xf");
 intupdate=namedParameterJdbcTemplate.update(sql, args);
 System.out.println("insert records : "+update);
 }
 ?
 /**
 * 使用具名参数时, 可以使用 update(String sql, SqlParameterSource paramSource) 方法进行更新操作
 * 1. SQL 语句中的参数名和类的属性一致!
 * 2. 使用 SqlParameterSource 的 BeanPropertySqlParameterSource 实现类作为参数.
 */
 @Test
 publicvoidtestNamedParameterJdbcTemplate2(){
 Stringsql="insert into t_user(user_name,nick_name)values(:userName,:nickName)";
 Useruser=newUser("Saodiseng","sds");
 SqlParameterSourceparamSource=newBeanPropertySqlParameterSource(user);
 namedParameterJdbcTemplate.update(sql, paramSource);
 }

附录-参考资料:

本文档参考了尚硅谷Spring教程和网上相关参考资料,感谢互联网的分享。

尚硅谷

[尚硅谷-官网] http://www.atguigu.com/

关于事务及传播行为

[MySQL的四种事务隔离级别] https://www.cnblogs.com/huanongying/p/7021555.html

[事务及其隔离级别] https://www.cnblogs.com/melody210218/p/7120559.html

[浅谈MySQL的事务级别] https://baijiahao.baidu.com/s?id=1572543485267773&wfr=spider&for=pc

[MySQL中的锁(表锁、行锁,共享锁,排它锁,间隙锁)] https://blog.csdn.net/soonfly/article/details/70238902

Tags:

本文暂时没有评论,来添加一个吧(●'◡'●)

欢迎 发表评论:

最近发表
标签列表