Spring事务和Aspects框架管理事务的用法
一.事务的介绍
1.1什么是事务
数据库事务( transaction)是访问并可能操作各种数据项的一个数据库操作序列,这些操作要么全部执行,要么全部不执行,是一个不可分割的工作单位。事务由事务开始与事务结束之间执行的全部数据库操作组成(百度百科)。
1.2事务的四大特性
1.原子性(Atomicity)
事务是数据库的逻辑工作单位,事务中包括的诸操作要么全做,要么全不做。
2.一致性(Consistency)
事务执行的结果必须是使数据库从一个一致性状态变到另一个一致性状态。一致性与原子性是密切相关的。
3.隔离性(Isolation)
各个事务之间的执行互不干扰。
4.持续性/永久性(Durability)
一个事务一旦提交,它对数据库中数据的改变就应该是永久性的。
二.事务的隔离级别以及导致的问题介绍
2.1隔离级别介绍
注:Mysql默认是REPEATABLE_READ,Oracle默认是READ_COMMITTED
2.2导致的问题介绍
2.2.1脏读
一个事务对数据进行了增删改查,但是未提交事务。另一个事物可以读取到未提交的数据,如果第一个事务进行了回滚,那么第二个事务就读到了脏数据。
例子:张三给了李四10000块现金让李四转账到张三的银行账户上,李四马上就转了10000块,张三此时也确认收到了转账记录,也看了账户确实多了10000块,但是该事务并没有提交,李四回到家之后立马回滚了事务,张三再次查看自己银行账户时,发现钱没了,显然李四用了高超的计算机技术蒙骗了张三.
2.2.2不可重复读
不可重复读,是指在数据库访问中,一个事务范围内两个相同的查询却返回了不同数据。
一种更易理解的说法是:在一个事务内,多次读同一个数据。在这个事务还没有结束时,另一个事务也访问该同一数据并修改数据。那么,在第一个事务的两次读数据之间。由于另一个事务的修改,那么第一个事务两次读到的数据可能不一样,这样就发生了在一个事务内两次读到的数据是不一样的,因此称为不可重复读,即原始读取不可重复。
例子:张三想从银行账户提现10000元,系统读到卡余额有20000元,此时张三兄弟李四正好也需要提现20000元,并且在张三提交事务前把20000元提现了,当张三提交事务时,系统读到卡上余额为0,提示余额不足。
2.2.3幻读
幻读,指的是当某个事务在读取或修改某个范围内的记录时,另外一个事务又在该范围内插入了新的记录,当之前的事务再次读取该范围的记录时,会产生幻行。
例子1:张三给全公司员工发工资时,第一个事务更新了数据库员工的工资表,随后入职了一个新的员工李四,第二个事务数据库管理员把员工信息加入了数据库(第一个事务还未提交),张三再次查看时发现李四这个员工还没有发工资,工资表还没有修改,张三很疑惑明明前一秒自己都修改了所有员工工资表,转眼看又多出来一个没有修改,就好象发生了幻觉一样.
例子2:张三老婆准备打印张三这个月的信用卡消费记录,经查询发现消费了两次共1000元,而这时张三刚按摩完准备结账,消费了1000元,这时银行记录新增了一条1000元的消费记录。当张三老婆将消费记录打印出来时,发现总额变为了2000元,这让张三老婆很诧异。
2.2.4串行话读
Serializable是最高的隔离级别,性能很低,一般很少用。在这级别下,事务是串行顺序执行的,不仅避免了脏读、不可重复读,还避免了幻读。
三.事务的传播行为
事务传播行为:是指一个事务方法A被另一个事务方法B调用时,这个事务A应该如何处理。事务A应该在事务B中运行还是另起一个事务,这个由事务A的传播行为决定。
掌握如下三个常用的传播行为
PROPAGATION_REQUIRED
PROPAGATION_REQUIRES_NEW
PROPAGATION_SUPPORTS
1) REQUIRED: spring默认传播行为, 方法在调用的时候,如果存在事务就是使用当前的事务,如果没有事务,则新建事务,
2) SUPPORTS:支持, 方法有事务可以正常执行,没有事务也可以正常执
行。
3)REQUIRES_NEW:方法需要一个新事务。 如果调用方法时,存在一个事务,则原来的事务暂停。 直到新事务执行完毕。 如果方法调用时,没有事务,则新建一个事务,在新事务执行代码。
四.Spring管理事务
3.1使用Spring的事务注解管理事务(掌握)
3.1.1Spring的事务注解管理事务的特点
@Transactional使用的特点:
1.spring框架自己提供的事务控制
2.适合中小型项目。
3.使用方便,效率高。
3.1.2实现步骤
1.在Spring.xml配置文件中声明事务内容
1)声明事务管理器
2)声明事务注解扫描器(告诉Spring使用了注解进行事务管理)
<!-- 声明事务管理器
Spring框架自带的事务管理器对象-->
<bean id="TransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!--指定数据源-->
<property name="dataSource" ref="MyDataSource"/>
</bean>
<!-- 开启事务注解驱动,告诉框架使用注解管理驱动,不然框架不知道,看不到事务注解,相当于不会执行事务操作-->
<tx:annotation-driven transaction-manager="TransactionManager"/>
2.在源代码中,需要添加事务的方法上加上@Transactional,注意只有public方法才可以
/**
* propagation指定传播行为,默认是REQUIRED
* isolation指定隔离级别,默认是DEFAULT
* readOnly是否只读,默认是false
* timeout设定超时时间,默认是-1没有,单位是秒
* rollbackfor指定回滚异常,也就是当遇到什么类型的异常是执行回滚,如果不声明则默认是RuntimeExeption异常时回滚
*
* 注意:@Transactional只能使用在public方法之上,private,protected等等都不想,
* 也就是只能给public方法加事务管理
* 加了@Transactional注解的方法,作为一个事务整体,方法里面的执行操作要么全部成功,要么全部失败
*/
// @Transactional(
// propagation = Propagation.REQUIRED,
// isolation = Isolation.DEFAULT,
// readOnly = false,
// timeout = 20,
// rollbackFor = {NullPointerException.class,NotEnoughException.class}
// )
@Transactional//一般使用默认就可以了,需要的时候再设置参数
@Override
public void buy(Integer id, Integer num) {
System.out.println("==buy方法开始执行==");
//购买商品的实体类
Goods goods = new Goods();
goods.setId(id);//购买商品的编号
goods.setAmount(num);//购买商品的数量
//插入更新销售表
Sale sale = new Sale();
sale.setGid(id);//销售商品的编号
sale.setNum(num);//销售商品的数量
saleDao.insertSale(sale);
//查询商品的库存
Goods sumgoods = goodsDao.selectByid(id);
//商品不存在
if(sumgoods==null){
throw new NullPointerException("商品不存在!");//抛出空指针异常
}
//商品库存不足
if(sumgoods.getAmount()<num){
throw new NotEnoughException("商品库存不足");//抛出自定义异常
}
//更新库存
goodsDao.updateGoods(goods);
System.out.println("==buy方法执行结束==");
}
3.2使用AspectJ的AOP配置管理事务(掌握)
3.2.1AspectJ的AOP配置管理事务的特点
声明式事务优缺点:
- 缺点: 理解难, 配置复杂。
- 优点: 代码和事务配置是分开的。 控制事务源代码不用修改。
能快速的了解和掌控项目的全部事务。 适合大型项目。
3.1.2实现步骤
1.在pom.xml中加入spring-aspects的依赖
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
<version>5.3.1</version>
</dependency>
2.在Spring.xml中声明事务的内容
1)声明事务管理器
2)给业务方法增加事务
3)声明切入点表达式,将事务和方法绑定
<beans xmlns="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" xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop
https://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!--声明式事务:不用写代码-->
<!--1.声明事务管理器对象-->
<bean id="TransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!--指定数据源-->
<property name="dataSource" ref="MyDataSource"/>
</bean>
<!--2.声明业务方法的事务属性(隔离级别,传播行为,超时)
id:给业务方法配置事务段代码起个名称,唯一值
transaction-manager:事务管理器的id
-->
<tx:advice id="ServiceAdvice" transaction-manager="TransactionManager">
<!--给具体的业务方法增加事务说明-->
<tx:attributes>
<!--name业务方法名称,
propagation:传播行为
isolation:隔离级别
read-only:只读
rollback-for:遇到什么异常执行事务-->
<!--给buy方法绑定事务-->
<tx:method name="buy" propagation="REQUIRED" isolation="DEFAULT" read-only="false" rollback-for="java.lang.NullPointerException com.bjpowernode.Exception.NotEnoughException"/>
<!--使用通配符*绑定事务-->
<!--给所有的add开头的方法绑定事务-->
<tx:method name="add*" propagation="REQUIRED"></tx:method>
</tx:attributes>
</tx:advice>
<!--什么切入点表达式,声明哪些包中的类的哪些方法参与事务-->
<aop:config>
<!--声明切入点表达式
expression:切入点表达式, 表示那些类和类中的方法要参与事务
id:切入点表达式的名称,唯一值
expression怎么写?
-->
<aop:pointcut id="ServicePointcut" expression="execution(* *..service..*.*(..))"/>
<!--关联切入点表达式和事务通知-->
<aop:advisor advice-ref="ServiceAdvice" pointcut-ref="ServicePointcut"/>
</aop:config>