Spring 事务特性
忘记 从哪转的了!!!
一、事务属性概述
在Spring中,事务属性描述了事务策略如何应用到方法上,事务属性包含5个方面:
① 传播行为
② 隔离级别
③ 回滚策略
④ 超时时间
⑤ 是否只读
二、事务的传播行为属性
当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。例如,方法可能继续在现有的事务中允许,也可能开启一个新事务,并在自己的事务中运行。
事务的传播行为可以由传播属性指定,Spring定义了7种类型的传播行为。其中最常用的是REQUIRED和REQUIRES_NEW。
这里写图片描述
事务的传播属性可以在@Transactional注解的propagation属性中定义。
举个例子:
/**
* 使用 @Transactional 指定事务方法
*/
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService
{
/**
* 1. 添加事务注解@Transactional。
* 2. 使用 propagation 指定事务的传播行为,
* 即当前事务方法被另外一个事务方法调用时,如何使用事务:
* 是使用调用方法的事务,还是开启一个新事务(使用自己的事务)。
* 3. propagation:
* ① 默认值是 Propagation.REQUIRED, 即使用调用方法的事务。
* ② 可以指定为 Propagation.REQUIRES_NEW, 即在调用方法的事务中开启一个新事务(自己的事务)。
*/
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void purchaseSigleBook(String acctName, String bookId)
{
...
}
}
三、事务的隔离级别
从理论上来说,事务应该彼此完全隔离,以避免并发事务所导致的问题。然而,那样会对性能产生极大的影响,因为事务必须按顺序运行。
在实际开发中,为了提升性能,事务会以较低的隔离级别运行。
事务的隔离级别可以通过隔离级别事务属性(isolation)指定。
Spring支持的事务隔离级别:
事务的隔离级别要得到底层数据库引擎的支持,而不是应用程序或者框架的支持。
Oracle 支持的2种事务隔离级别:READ_COMMIT。
MySQL 支持4种事务隔离级别:READ-UNCOMMITTED、READ-COMMITTED、REPEATABLE-READ 和 SERIALIZABLE。
这里写图片描述
用 @Transactional注解声明式的管理事务时可以在 @Transactional 的isolation属性中设置隔离级别。
举个例子:
/**
1. 使用 @Transactional 指定事务方法
*/
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService
{
/**
* 1. 添加事务注解@Transactional。
* 2. 使用isolation属性指定事务的隔离级别,最常用的的取值是: Isolation.READ_COMMITTED。
*/
@Transactional(isolation = Isolation.READ_COMMITTED)
public void purchaseSigleBook(String acctName, String bookId)
{
...
}
}
四、事务的回滚属性
默认情况下只有未检查异常( RuntimeException 和 Error 类型的异常)会导致事务回滚,而检查异常不会。
事务回滚的规则可以通过 @Transactional注解的 rollbackFor和 noRollbackFor属性来定义,这两个属性被声明为Class[] 类型的,因此可以为这两个属性指定多个异常类。
① rollbackFor:指定遇到哪些异常时必须进行回滚。
② noRollbackFor:指定遇到哪些异常时,必须不回滚。
举个例子:
/**
* 使用 @Transactional 指定事务方法
*/
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService
{
/**
* 1. 添加事务注解@Transactional。
* 2. 使用rollbackFor 和 noRollbackFor指定事务遇到哪些异常时进行强制回滚或者强制不回滚。
* 默认情况下,Spring声明事务对所有运行时异常进行回滚.通常取默认值。
*/
@Transactional(noRollbackFor = {UserAccounException.class})
public void purchaseSigleBook(String acctName, String bookId)
{
...
}
}
五、事务的只读属性
如果一个事务只读取数据但不修改数据,数据库引擎可以对这个事务进行优化。
只读事务属性:表示这个事务只读取数据但不更新数,这样可以帮助数据库进行优化事务。
只读属性可以在 @Transactional注解中readOnly属性中定义,其取值只有true和false。
举个例子:
/**
* 使用 @Transactional 指定事务方法
*/
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService
{
/**
* 1. 添加事务注解@Transactional。
* 2. 使用 readOnly 只读事务的只读属性,表示这个事务是否只读取数据而不修改数据。
*/
@Transactional(readOnly = true)
public void purchaseSigleBook(String acctName, String bookId)
{
...
}
}
六、事务的超时属性
由于事务可以在行和表上获得锁,因此长期运行的事务会占用资源,并对整体性能产生影响。
超时事务属性:事务在强制回滚之前可以保持多久,这样可以防止长期运行的事务占用资源。
超时属性可以在 @Transactional注解中timeout属性中定义,其单位为秒。
举个例子:
/**
* 使用 @Transactional 指定事务方法
*/
@Service("bookShopService")
public class BookShopServiceImpl implements BookShopService
{
/**
* 1. 添加事务注解@Transactional。
* 2. 使用 timeout 指定事务的超时属性,表示在强制回滚之前,这个事务能运行多久,单位是秒。
*/
@Transactional(timeout = 3)
public void purchaseSigleBook(String acctName, String bookId)
{
...
}
}