解决org.springframework.dao.DataIntegrityViolationException: could not execute statement; SQL [n/a];

,在同一代码块内逻辑删除数据,在添加一条数据时,在同一事物内,出现异常
org.springframework.dao.DataIntegrityViolationException:
 could not execute statement; SQL [n/a]; 
constraint [uni_payuuid]; nested exception is org.hibernate.exception.ConstraintViolationException: could not execute statement
需要在不同的事物内执行,定义事物策略,开启新的事物

Spring事务管理高层抽象主要包括3个接口

  --Platform TransactionManager 事务管理器(提交、回滚事务)

     Spring为不同的持久化框架提供了不同的Platform TransactionManager接口实现。如:

        使用Spring JDBC或iBatis进行持久化数据时使用DataSourceTransactionManager

        使用Hibernate3.0版本进行持久化数据时使用HibernateTransactionManager

  --TransactionDefinition 事务定义信息(隔离、传播、超时、只读)

        脏读:一个事务读取了另一个事务改写但还未提交的数据,如果这些数据被回滚,则读到的数据是无效的。

        不可重复读:在同一事务中,多次读取同一数据返回的结果有所不同。

        幻读:一个事务读取了几行记录后,另一个事务插入一些记录,幻读就发生了。再后来的查询中,第一个事务就会发现有些原来没有的记录。

        事务隔离级别:(五种)

  •     DEFAULT--使用后端数据库默认的隔离级别(Spring中的选择项)
  •     READ_UNCOMMITED--允许你读取还未提交的改变了的数据。可能导致脏、幻、不可重复读
  •     READ_COMMITTED--允许在并发事务已经提交后读取。可防止脏读,但幻读和不可重复读仍可发生
  •     REPEATABLE_READ--对相同字段的多次读取是一致的,除非数据被事务本身改变。可防止脏、不可重复读,但幻读仍可能发生
  •     SERIALIZABLE--完全服从ACID的隔离级别,确保不发生脏、幻、不可重复读。这在所有的隔离级别中是最慢的,它是典型的通过完全锁定在事务中涉及的数据表来完成的

    其中,MySQL默认采用REPEATABLE_READ隔离级别;Oracle默认采用READ_COMMITTED隔离级别

        事务传播行为:(七种)

  •     REQUIRED--支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
  •     SUPPORTS--支持当前事务,如果当前没有事务,就以非事务方式执行。
  •     MANDATORY--支持当前事务,如果当前没有事务,就抛出异常。
  •     REQUIRES_NEW--新建事务,如果当前存在事务,把当前事务挂起。
  •     NOT_SUPPORTED--以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。
  •     NEVER--以非事务方式执行,如果当前存在事务,则抛出异常。
  •     NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与REQUIRED类似的操作。拥有多个可以回滚的保存点,内部回滚不会对外部事务产生影响。只对DataSourceTransactionManager有效

  --TransactionStatus 事务具体运行状态

 

解决org.springframework.dao.DataIntegrityViolationException: could not execute statement; SQL [n/a];_第1张图片

 

  a.编程式事务管理(基于Java编程控制,很少使用)--见demo1包

       利用TransactionTemplate将多个DAO操作封装起来

  *b.声明式事务管理(基于Spring的AOP配置控制)

       -基于TransactionProxyFactoryBean的方式.(很少使用)--见demo2包

            需要为每个进行事务管理的类,配置一个TransactionProxyFactoryBean进行增强.

       -基于XML配置(经常使用)--见demo3包

            一旦配置好之后,类上不需要添加任何东西。

            如果Action作为目标对象切入事务,需要在元素里添加proxy-target-class="true"属性。原因是通知Spring框架采用CGLIB技术生成具有事务管理功能的Action类。

       -基于注解(配置简单,经常使用)--见demo4包

            在applicationContext.xml中开启事务注解配置。(applicationContext.xml中只需定义Bean并追加以下元素)

?

            在目标组件类中使用@Transactional,该标记可定义在类前或方法前。

你可能感兴趣的:(Sring)