开启事务
<tx:annotation-driven transaction-manager="defaultTransactionManager" proxy-target-class="true" />
脏读:一个事务读取到另一事务未提交的更新数据
不可重复读:在同一事务中, 多次读取同一数据返回的结果有所不同, 换句话说, 后续读取可以读到另一事务已提交的更新数据. 相反, "可重复读"在同一事务中多次读取数据时, 能够保证所读数据一样, 也就是后续读取不能读到另一事务已提交的更新数据
幻读:一个事务读到另一个事务已提交的insert数据
事务隔离级别:
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
读取未提交数据(会出现脏读, 不可重复读) 基本不使用
@Transactional(isolation = Isolation.READ_COMMITTED)
读取已提交数据(会出现不可重复读和幻读)
@Transactional(isolation = Isolation.REPEATABLE_READ)
可重复读(会出现幻读)
@Transactional(isolation = Isolation.SERIALIZABLE)
串行化
MYSQL: 默认为REPEATABLE_READ
SQLSERVER: 默认为READ_COMMITTED
事物传播行为介绍:
@Transactional(propagation=Propagation.REQUIRED)
有事务, 加入事务, 没有则新建一个(默认情况下)
@Transactional(propagation=Propagation.NOT_SUPPORTED)
不开启事务
@Transactional(propagation=Propagation.REQUIRES_NEW)
始终创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务
@Transactional(propagation=Propagation.MANDATORY)
必须在一个已有的事务中执行,否则抛出异常
@Transactional(propagation=Propagation.NEVER)
必须在一个没有的事务中执行,否则抛出异常(与Propagation.MANDATORY相反)
@Transactional(propagation=Propagation.SUPPORTS)
其他bean调用这个方法时,在其他bean中有事务,就用事务.没有声明事务,就不用事务.
readOnly:当前事务是否为只读事务,true表示只读,false则表示可读写,默认值为false。
如果为true,事务方法中有写操作则抛出异常。例如:@Transactional(readOnly=true)
rollbackFor:事务回滚 如:@Transactional(rollbackFor=RuntimeException.class),@Transactional(rollbackFor={RuntimeException.class, Exception.class})
————————————————————
1、@Transactional 只能被应用到public方法上, 对于其它非public(如:protected、private)的方法,如果标记了@Transactional也不会报错,但方法没有事务功能.
2、spring负责数据库的打开,提交,回滚,默认遇到运行期例外(throw new RuntimeException("×××");)会回滚,而遇到需要捕获的例外(throw new Exception("×××");)不会回滚。要想所有异常都回滚,要加上 @Transactional( rollbackFor={Exception.class,其它异常})。
————————————————————
建议是在具体的类(或类的方法)上使用 @Transactional 注解,而不要使用在类所要实现的任何接口上。
当然可以在接口上使用 @Transactional 注解,但是这将只能当你设置了基于接口的代理时它才生效。
因为注解是 不能继承 的,这就意味着如果你正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,而且对象也将不会被事务代理所包装!
<tx:annotation-driven transaction-manager="defaultTransactionManager" proxy-target-class="true" />
脏读:一个事务读取到另一事务未提交的更新数据
不可重复读:在同一事务中, 多次读取同一数据返回的结果有所不同, 换句话说, 后续读取可以读到另一事务已提交的更新数据. 相反, "可重复读"在同一事务中多次读取数据时, 能够保证所读数据一样, 也就是后续读取不能读到另一事务已提交的更新数据
幻读:一个事务读到另一个事务已提交的insert数据
事务隔离级别:
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
读取未提交数据(会出现脏读, 不可重复读) 基本不使用
@Transactional(isolation = Isolation.READ_COMMITTED)
读取已提交数据(会出现不可重复读和幻读)
@Transactional(isolation = Isolation.REPEATABLE_READ)
可重复读(会出现幻读)
@Transactional(isolation = Isolation.SERIALIZABLE)
串行化
MYSQL: 默认为REPEATABLE_READ
SQLSERVER: 默认为READ_COMMITTED
事物传播行为介绍:
@Transactional(propagation=Propagation.REQUIRED)
有事务, 加入事务, 没有则新建一个(默认情况下)
@Transactional(propagation=Propagation.NOT_SUPPORTED)
不开启事务
@Transactional(propagation=Propagation.REQUIRES_NEW)
始终创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务
@Transactional(propagation=Propagation.MANDATORY)
必须在一个已有的事务中执行,否则抛出异常
@Transactional(propagation=Propagation.NEVER)
必须在一个没有的事务中执行,否则抛出异常(与Propagation.MANDATORY相反)
@Transactional(propagation=Propagation.SUPPORTS)
其他bean调用这个方法时,在其他bean中有事务,就用事务.没有声明事务,就不用事务.
readOnly:当前事务是否为只读事务,true表示只读,false则表示可读写,默认值为false。
如果为true,事务方法中有写操作则抛出异常。例如:@Transactional(readOnly=true)
rollbackFor:事务回滚 如:@Transactional(rollbackFor=RuntimeException.class),@Transactional(rollbackFor={RuntimeException.class, Exception.class})
rollbackForClassName:如:@Transactional(rollbackForClassName="RuntimeException"),@Transactional(rollbackForClassName={"RuntimeException","Exception"})
noRollbackFor:事务不回滚,与rollbackFor相反
noRollbackForClassName:事务不回滚,与rollbackForClassName相反
————————————————————
1、@Transactional 只能被应用到public方法上, 对于其它非public(如:protected、private)的方法,如果标记了@Transactional也不会报错,但方法没有事务功能.
2、spring负责数据库的打开,提交,回滚,默认遇到运行期例外(throw new RuntimeException("×××");)会回滚,而遇到需要捕获的例外(throw new Exception("×××");)不会回滚。要想所有异常都回滚,要加上 @Transactional( rollbackFor={Exception.class,其它异常})。
————————————————————
建议是在具体的类(或类的方法)上使用 @Transactional 注解,而不要使用在类所要实现的任何接口上。
当然可以在接口上使用 @Transactional 注解,但是这将只能当你设置了基于接口的代理时它才生效。
因为注解是 不能继承 的,这就意味着如果你正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,而且对象也将不会被事务代理所包装!