1. 事务
1.1 4个基本特性(ACID)
4个基本属性:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability):
- 原子性:事务内容要么全部执行成功,要么全部执行失败,成功则写入数据库,失败则对数据库不产生影响;
- 隔离性:用户并发访问时会同时存在多个事务,多个并发事务之间相互隔离;
- 一致性:事务执行前和执行后状态保持一致,比如A向B转账,A扣钱后B必定收到等额入账,且总数不变;
- 持久性:事务完成后,事务对数据库的所有更新执行都永久性保存到数据库,不能再被回滚;
1.2 事务并发问题
多个事务并发时可能出现以下并发问题:
- 脏读:事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据;
- 不可重复读:事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据的结果不一致;
- 幻读:系统管理员A将数据库中所有学生的成绩从具体分数改为ABCDE等级,但是系统管理员B就在这个时候插入了一条具体分数的记录,当系统管理员A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。
小结:不可重复读的和幻读很容易混淆,不可重复读侧重于修改,幻读侧重于新增或删除。解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表
1.3 事务隔离级别
为了处理事务并发问题,可以为事务设置不同的隔离级别。
MySQL数据库支持4种隔离级别:
- Read Uncommitted(读未提交):最低级别,任何情况都可能发生;
- Read Committed(读已提交):可避免脏读的发生;
- Repeatable Read(可重复读):可避免脏读、不可重复读的发生;
- Serializable(串行化):最高级别,避免脏读、不可重复读、幻读的发生;
需要注意一下几点:
- 四种隔离级别最高的是Seralizable级别,最低的是Read Uncommitted级别,级别越高执行效率就越低;
- 隔离级别的设置只对当前链接有效,对JDBC操作数据库来说,一个Connection对象相当于一个链接;
- Mysql的默认隔离级别是:可重复读(Repeatable Read);
- Oracle只支持串行化(Seralizable)级别和读已提交(Read committed)级别,默认是读已提交(Read Committed)级别。
- 采用最高事务隔离级别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 常见的事务管理器
- org.springframework.jdbc.datasource.DataSourceTransactionManager : 应用程序只需要处理一个数据源,并且通过JDBC存取;
- org.springframework.transaction.jta.JtaTransactionManager : 在JavaEE应用服务器上用JTA(Java Transaction API)进行事务管理;
- 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 小需求
关于事务的小需求:用户购买指定图书,图书有一定库存,用户有一定的金钱,每次只能购买一本图书,购买需要花费对应书价的金钱,购买方法使用事务管理,过程分为三步:
- 根据书号ISBN号查询图书价格;
- 扣减图书库存,每次只能扣减一本,库存不足时抛出异常;
- 扣减用户账户余额,扣减数目与图书价格相同,余额不足时购买失败并抛出异常。
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本,或者,一本都买不了。
需要设置:
- 用户qiaofeng账户余额为50元,图书IB1234售价为50元,库存为20;图书IB1235售价为100元,库存为20;
- 用户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种隔离级别(级别右低变高):
- Read Uncommitted(读未提交):最低级别,任何情况都可能发生;
- Read Committed(读已提交):可避免脏读的发生;
- Repeatable Read(可重复读):可避免脏读、不可重复读的发生;
- Serializable(串行化):最高级别,避免脏读、不可重复读、幻读的发生;
关于事务的隔离级别,需要注意的是:
- 从理论上来说,事务应该彼此完全隔离,以避免并发事务所导致的问题,然而,那样会对性能产生极大的影响, 因为事务必须按顺序运行;
- 在实际开发中,为了提升性能,事务会以较低的隔离级别运行;
- MySQL数据库支持4种事务隔离级别,Oracle数据库只支持其中2种:READ_COMMITED , SERIALIZABLE;
- 事务的隔离级别要得到底层数据库引擎的支持,而不是应用程序或者框架的支持。
事务的隔离级别可以通过隔离事务属性指定,指定属性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
本文暂时没有评论,来添加一个吧(●'◡'●)