Spring 事务

   

spring事务:

什么是事务:

事务逻辑上的一组操作,组成这组操作的各个逻辑单元,要么一起成功,要么一起失败.

   

事务特性(4种):

原子性 atomicity:强调事务的不可分割.

一致性 consistency:事务的执行的前后数据的完整性保持一致.

隔离性 isolation:一个事务执行的过程中,不应该受到其他事务的干扰

持久性(durability :事务一旦结束,数据就持久到数据库

   

如果不考虑隔离性引发安全性问题:

脏读 : 一个事务读到了另一个事务的未提交的数据

不可重复读 : 一个事务读到了另一个事务已经提交的 update 的数据导致多次查询结果不一致.

虚幻读 : 一个事务读到了另一个事务已经提交的 insert 的数据导致多次查询结果不一致.

   

解决读问题: 设置事务隔离级别(5种)

DEFAULT 这是一个PlatfromTransactionManager默认的隔离级别,使用数据库默认的事务隔离级别.

未提交读read uncommited :脏读,不可重复读,虚读都有可能发生

已提交读 read commited:避免脏读。但是不可重复读和虚读有可能发生

可重复读 repeatable read :避免脏读和不可重复读.但是虚读有可能发生.

串行化的 serializable :避免以上所有读问题.

   

Mysql 默认:可重复读

Oracle 默认:读已提交

   

这里写图片描述

read uncommited是最低的事务隔离级别,它允许另外一个事务可以看到这个事务未提交的数据。

read commited保证一个事物提交后才能被另外一个事务读取。另外一个事务不能读取该事物未提交的数据。

repeatable read:这种事务隔离级别可以防止脏读,不可重复读。但是可能会出现幻象读。它除了保证一个事务不能被另外一个事务读取未提交的数据之外还避免了以下情况产生(不可重复读)。

serializable这是花费最高代价但最可靠的事务隔离级别。事务被处理为顺序执行。除了防止脏读,不可重复读之外,还避免了幻象读(避免三种)。

   

  • 事务的传播行为

    PROPAGATION_XXX :事务的传播行为

  • 保证同一个事务中

    PROPAGATION_REQUIRED 支持当前事务,如果不存在 就新建一个(默认)

    PROPAGATION_SUPPORTS 支持当前事务,如果不存在,就不使用事务

    PROPAGATION_MANDATORY 支持当前事务,如果不存在,抛出异常

  • 保证没有在同一个事务中

    PROPAGATION_REQUIRES_NEW 如果有事务存在,挂起当前事务,创建一个新的事务

    PROPAGATION_NOT_SUPPORTED 以非事务方式运行,如果有事务存在,挂起当前事务

    PROPAGATION_NEVER 以非事务方式运行,如果有事务存在,抛出异常

    PROPAGATION_NESTED 如果当前事务存在,则嵌套事务执行

   

事务 :1、编程式事务管理(嵌入编码中,灵活性高,不宜维护)

2、声明式事务管理(利用注解和xml配置文件,将业务代码和事务隔离,低侵入性。相较于编程式事务缺少灵活性)

声明式事务配置:

①注解式配置@Transction

   

Transactional注解不回滚

***最有可能的问题就是:spring.xml 和spring-mvc.xml都开启了注解扫描,因为spring和springmvc相当于父子容器,springmvc.xml扫描到@service里的@Transction在mvc里面是没有经过配置的。所以建议只在springmvc.xml做一些web相关的配置即可。

1、检查你方法是不是public的

2、你的异常类型是不是unchecked异常

如果我想check异常也想回滚怎么办,注解上面写明异常类型即可

@Transactional(rollbackFor=Exception.class)

类似的还有norollbackFor,自定义不回滚的异常

3、数据库引擎要支持事务,如果是MySQL,注意表要使用支持事务的引擎,比如innodb,如果是myisam,事务是不起作用的

4、是否开启了对注解的解析

<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

5、spring是否扫描到你这个包,如下是扫描到org.test下面的包

<context:component-scan base-package="org.test" ></context:component-scan>

6、检查是不是同一个类中的方法调用(如a方法调用同一个类中的b方法)

7、异常是不是被你catch住了

   

/**

* 事物注解

* 使用propagation 指定事物的传播行为,即当前事物方法被另一个事物方法调用时如何使用事物,默认取值为REQUIRED,即使用调用方法的事物

* REQUIRES_NEW:使用自己的事物,调用的事物方法的事物被挂起。

* isolation:指定事务的隔离级别, 最常用的取值为:READ_COMMITTED,读以提交

* noRollbackFor:对这个异常不进行回滚 通常情况下取默认值即可。

* rollbackFor:对这个异常进行回滚

* readOnly:只读事务,只能读取数据,可以帮助数据库引擎优化。如果是只读,应设置:readOnly=true

* timeout:指定强制回滚的时间,单位秒,如该方法执行了5秒,而该属性设置的是2秒,如果到2秒了,该方法还没有执行完,该事务也会对该方法进行强制回滚。

*/

@Transactional(propagation=Propagation.REQUIRED,isolation=Isolation.READ_COMMITTED,

noRollbackFor={UserException.class},noRollbackForClassName="UserException",

rollbackFor={UserException.class},rollbackForClassName="UserException",

readOnly=false,timeout=100)

public void updatUser(){

System.out.println("需要用到事务的方法");

}

   

②Aop方式

aop相关的jar

   

   

配置完成后,切入点的类、方法,都会加上事务。

   

   

配置完这些后,只是说spring-tx会拦截expression="execution…."路径下的类和方法。

   

但是拦截之后,并不会处理,所以我们需要建立一个专门的异常处理类来处理这些拦截到的异常。

   

配置过程中遇到的问题

***

普通的controller类,如果使用ResponseBody,像前端写值,那么很肯能乱码,所以需要在@RequestMapping(value="a",produces="application/json;charset=utf-8")

   

***

异常处理类中,因为无序配置@RequestMapping(),所以也不能配置produces。

经测试:如果异常处理类中返回一个字符串,将会乱码。

但如果返回ResponseEntity类,将不会有乱码。

posted @ 2020-10-18 20:04  黑质白章  阅读(103)  评论(0编辑  收藏  举报