提起spring事务,就会让人联想起四大基本特征,五个隔离级别,七大传播特性。相信大多数人都知道这些东西,但是知道是一回事情,能用好真的是另一回事了。在使用Spring事务的时候,我曾遇到过几个比较严肃的问题,在这里我做一个自我总结。

问题一、 propagation.NESTED和propagation.REQUIRED_NEW有什么区别?

  当调用方不存在事务的时候,两者的效果是一致的。所以这里讨论问题的前提是调用方存在事务。PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 "内部" 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行. 
另一方面, PROPAGATION_NESTED 开始一个 "嵌套的" 事务, 它是已经存在事务的一个真正的子事务. 潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint. 潜套事务是外部事务的一部分, 只有外部事务结束后它才会被提交. 
由此可见, PROPAGATION_REQUIRES_NEW 和 PROPAGATION_NESTED 的最大区别在于, PROPAGATION_REQUIRES_NEW 完全是一个新的事务, 而 PROPAGATION_NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 这个规则同样适用于 roll back.

问题二、 @Transactional为什么会失效?

  1.调用方和被调用方属于同一个component,被调用方的 @Transacational注解无效

  

package com.transacational;

import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; /**
* Created by chenqimiao on 17/10/31.
*/
@Component
public class Service { public void test1(){
test2();
} @Transactional//此处的注解无效
public void test2(){ }
}

  2.被调用方不是一个public方法,被调用方的 @Transacational注解无效  

@Component
public class Service { @Resource
private Service1 service1; public void test1(){
test2();
service1.test3();
} @Transactional//1.此处的注解无效
public void test2(){ }
}
package com.transacational;

import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional; /**
* Created by chenqimiao on 17/10/31.
*/
@Component
public class Service1 { @Transactional//2.此处注解无效
protected void test3(){ }
}

  

  3.未开启事务开关,如:在SpringBoot中,启动类未使用 @EnableTransactionManagement

问题三、 如何理解@Transactional的超时时间?

   timeout是一个供开发者设置超时时间的属性。默认值-1,超时时间由具体的sql系统决定。  

/**
* Created by chenqimiao on 17/10/31.
*/
@Component
public class Service3 { @Resource
private AdminInfoDoMapper adminInfoDoMapper;
@Transactional(timeout = 4)//并不会超时
public void test4(){ adminInfoDoMapper.selectNameById(1);
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

超时时间具体的定义:事务开始(在该方法第一句代码执行之前)到最后一个Statement执行完毕

所以象下面这样写,事务就会超时

@Component
public class Service3 { @Resource
private AdminInfoDoMapper adminInfoDoMapper;
@Transactional(timeout = 4)
public void test4(){
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
adminInfoDoMapper.selectNameById(1); }
}

问题四、 @Transactional默认的回滚策略?

默认情况下,只有当RuntimeException或其子类的异常被事务捕获之后,事务才会回滚,如果要让事务能够回滚所有异常,必须手动指定  @Transactional(rollbackFor=Exception.class)  ,这样继承Exception的子类或者Exception本身都可以让事务回滚。

最新文章

  1. 10. JEB1.5 插件编写二
  2. 转:ASP.NET 使用Ajax
  3. sphinx配置文件sphinx.conf参数详细说明
  4. CocurrentHashMap和Hashtable的区别
  5. HDU 3555 数位dp入门
  6. jquery()的三种$()
  7. Android NDK 开发(三)--常见错误锦集合Log的使用【转】
  8. Create,Insert
  9. Maximum number of WAL files in the pg_xlog directory (2)
  10. 如何搭建Struts2环境
  11. 部署K2 Blackpearl流程时出错(与基础事务管理器的通信失败或Communication with the underlying transaction manager has failed.
  12. 类 ArrayBlockingQueue<E>(一个由数组支持的有界阻塞队列。)
  13. linux-telnet服务配置
  14. UIButton 头文件常见属性和方法
  15. js判断undefined类型,undefined,null,NaN的区别
  16. redhat在线安装chrome浏览器
  17. Luogu [USACO08OPEN]寻宝之路Clear And Present Danger
  18. Python元祖
  19. System.nanoTime与System.currentTimeMillis的区别(转)
  20. File的创建

热门文章

  1. 泛型在Web中的作用
  2. WEB项目的部署结构
  3. 随便讲讲我对于svn和git的想法
  4. NSTimer、CADisplayLink 内存泄漏
  5. struts2一个和多个文件上传及下载
  6. Python 编程基础之高阶函数篇(一)
  7. pycharm(windows)安装及其设置中文菜单
  8. 为什么说程序员都应该玩一玩GitHub
  9. Quartz入门案例与介绍(与spring整合)
  10. Yii 2.0 数据库操作总结