NEVER

不使用事务,如果当前事务存在,则抛出异常

验证:

@Service
public class PrService {
@Autowired
PrDao dao;
@Transactional
public void savea() {
dao.a();//保存第一条数据
saveb();
}
@Transactional(propagation = Propagation.NEVER)
private void saveb() {
dao.b();//保存第二条数据
int i=1/0;
}
}

写一个controller调用这个savea方法,页面看到的是/ by zero,数据库中两条数据都没有插入进去,都回滚了。照说设置了Propagation.NEVER,应该saveb方法根本执行不了,报never的错。为什么执行了呢?

因为是直接调用的saveb(),不是从代理对象上调用的方法,改成这样:

@Service
public class PrService {
@Autowired
PrDao dao; @Autowired
PrService prService; @Transactional
public void savea() {
dao.a();//保存第一条数据
prService.saveb();
} @Transactional(propagation = Propagation.NEVER)
private void saveb() {
dao.b();//保存第二条数据
int i=1/0;
}
}

注意这里是  prService.saveb()和前面不同,但是居然在  dao.b(); 这一行报了一个空指针的错误,这就诡异了,这个dao明明上a方法里面还有值的,你知道是为什么吗?

原来b方法的修饰符是private,导致代理对象无法继承这个方法。改成public

public void saveb()

再次测试,看到期待的报错了:

Existing transaction found for transaction marked with propagation 'never'

MANDATORY

当前存在事务,则加入当前事务,如果当前事务不存在,则抛出异常。

这里把MANDATORY放在b方法上,就要把a方法上的transactional注解去掉,才能看到报错。

@Service
public class PrService {
@Autowired
PrDao dao; @Autowired
PrService prService; public void savea() {
dao.a();//保存第一条数据
prService.saveb();
} @Transactional(propagation = Propagation.MANDATORY)
public void saveb() {
dao.b();//保存第二条数据
}
}

或者直接把MANDATORY放在a方法上也会报错:

@Service
public class PrService {
@Transactional(propagation = Propagation.MANDATORY)
public void savea() {
}
}

报错如下

No existing transaction found for transaction marked with propagation 'mandatory'

REQUIRES_NEW

创建一个新事务,如果存在当前事务,则挂起该事务。

可以理解为设置事务传播类型为REQUIRES_NEW的方法,在执行时,不论当前是否存在事务,总是会新建一个事务。

现在写一个demo验证,a方法调用b方法,希望b方法里面异常了且事务回滚,a方法里面不回滚

下面验证:

同样为了在避免事务失效,在代理对象上调用方法,将被调用的方法放到一个单独的类中:

@Service
public class PrService {
@Autowired
PrDao dao; @Autowired
SaveBService saveBService; @Transactional
public void savea() {
dao.a();//插入第一条数据
saveBService.saveb();
}
}
@Service
public class SaveBService {
@Autowired
PrDao dao; @Transactional(propagation = Propagation.REQUIRES_NEW)
public void saveb() {
dao.b();//插入第二条数据
int i = 1/0;
}
}

这里PrService的savea方法调用SaveBService的saveb方法,写一个controller访问savea方法,页面看到一个异常 / by zero,再检查数据库,两条数据都有没有插入进去。这是为什么呢?看上去第二个方法已经写到独立的类中了。

原来是因为b方法抛出的异常a方法没有捕获,又抛出去了,当然要回滚了。写个try catch就行:

    @Transactional
public void savea() {
dao.a();//插入第一条数据
try {
saveBService.saveb();
}catch (Exception e) {
System.out.println("some error");
}
}

再次测试,第二条数据没有插入,第一条数据插入到数据库了,页面也看不到报错了。

 
 

spring事务传播的Propagation.REQUIRES_NEW以及NEVER MANDATORY验证,及其失效的诡异问题的更多相关文章

  1. spring 事务传播(Propagation)

    propagation 一共有以下几种选项: 1. REQUIRED(默认): 使用当前的事务,如果当前没有事务,则自己新建一个事务,子方法必须运行在一个事务中:如果当前存在事务,则加入这个事务,成为 ...

  2. spring事务传播行为之使用REQUIRES_NEW不回滚

    最近写spring事务时用到REQUIRES_NEW遇到一些不回滚的问题,所以就记录一下. 场景1:在一个服务层里面方法1和方法2都加上事务,其中方法二设置上propagation=Propagati ...

  3. spring事务传播机制实例讲解

    http://kingj.iteye.com/blog/1680350   spring事务传播机制实例讲解 博客分类:   spring java历险     天温习spring的事务处理机制,总结 ...

  4. spring 事务传播机制

    spring 事务 传播机制 描述的 事务方法直接相互调用,父子事物开启,挂起,回滚 等的处理方式. 绿色的 那几个 我认为比较重要. 1 , @Transactional(propagation=P ...

  5. spring 事务传播行为实例分析

    Spring事务传播行为: spring特有的事务传播行为,spring支持7种事务传播行为,确定客户端和被调用端的事务边界(说得通俗一点就是多个具有事务控制的service的相互调用时所形成的复杂的 ...

  6. Spring 事务传播行为的使用

                                                                                                        ...

  7. 理解 spring 事务传播行为与数据隔离级别

    事务,是为了保障逻辑处理的原子性.一致性.隔离性.永久性. 通过事务控制,可以避免因为逻辑处理失败而导致产生脏数据等等一系列的问题. 事务有两个重要特性: 事务的传播行为 数据隔离级别 1.事务传播行 ...

  8. spring事务传播行为的思考

    1.问题 @TransactionConfiguration(transactionManager = "txManager", defaultRollback = false) ...

  9. spring事务传播行为讲解转载

    https://segmentfault.com/a/1190000013341344 前言 Spring在TransactionDefinition接口中规定了7种类型的事务传播行为.事务传播行为是 ...

  10. spring事务传播属性和隔离级别

    猫咪咪的Java世界 spring事务传播属性和隔离级别 博客分类: Spring java编程   1 事务的传播属性(Propagation) 1) REQUIRED ,这个是默认的属性 Supp ...

随机推荐

  1. kg record

    参考资料: https://www.kaggle.com/youngyang/a-different-eda-based-on-qlib-en

  2. 后端008_配置Security登录授权过滤器

    ------------恢复内容开始------------ 现在我们就可以去进行springscurity的配置了.首先我们新建一个配置类.然后该类需要添加@Configuration注解,然后还要 ...

  3. js对象深拷贝方法

    JSON.stringify()是目前前端开发过程中最常用的深拷贝方式, 原理是把有个对象序列化成为一个 JSON 字符串,将对象的内容转换成字符串的形式再保存到磁盘上, 再用 JSON.parse( ...

  4. imx6ull调试记录——开发环境搭建

    搭建开发环境之网络环境 代码编译环境准备 换源 sudo cp /etc/apt/sources.list /etc/apt/sources.list.bak sudo vim /etc/apt/so ...

  5. Web 开发的常规流程

    Web 开发的常规流程 What is the Web? 简单地说,网络是一个遍布全球的网络,它连接大量设备并允许它们相互通信 Internet 上的网站托管在称为服务器的设备上,当您与 Intern ...

  6. vulnhub靶场之WIRELESS: 1

    准备: 攻击机:虚拟机kali.本机win10. 靶机:Wireless: 1,下载地址:https://download.vulnhub.com/wireless/Mystiko-Wireless. ...

  7. ColorWell - web 颜色代码取色工具,Mac 上的优秀调色板

    ColorWell 是 Mac 上的一款非常优秀的颜色取色工具,她具有历史记录.调色板同步等功能,非常适合 web 或 App 开发人员使用 下载 ► ColorWell 下载安装 ⇲ 详细介绍 美丽 ...

  8. Thread 线程中的 Synchronized block and lock

    Thread Definition of Synchronized Synchronized block in java are marked with the synchronized keywor ...

  9. 中心极限定理的模拟—R实现

    中心极限定理,是指概率论中讨论随机变量序列部分和分布渐近于正态分布的一类定理.这组定理是数理统计学和误差分析的理论基础,指出了大量随机变量近似服从正态分布的条件.它是概率论中最重要的一类定理,有广泛的 ...

  10. pysimplegui之窗口大小,位子,主题等属性修改

    重点 1finalize()或Window参数finalize=True 调用以强制窗口通过初始化的最后阶段.这将导致 tkinter 资源被分配,以便它们可以被修改.这也会导致您的窗口出现.如果您不 ...