以下事务传播机制都用这个例子解释:有A方法,B方法两个方法,A调用B,方法A是一个事务的方法,方法A执行过程中调用了方法B,那么方法B有无事务以及方法B对事务的要求不同都会对方法A的事务具体执行造成影响,同时方法A的事务对方法B的事务执行也有影响,这种影响具体是什么就由两个方法所定义的事务传播类型所决定
这些传播机制都是修饰被调用者的,也就是这里的B
1、REQUIRED
(Spring默认的事务传播类型 required:需要、依赖、依靠):如果当前没有事务,则自己新建一个事务,如果当前存在事务则加入这个事务
当A调用B的时候:如果A中没有事务,B中有事务,那么B会新建一个事务;如果A中也有事务、B中也有事务,那么B会加入到A中去,变成一个事务,这时,要么都成功,要么都失败。(假如A中有2个SQL,B中有两个SQL,那么这四个SQL会变成一个SQL,要么都成功,要么都失败)
1.1正常情况,会回滚
@Transactional(propagation = Propagation.REQUIRED)
public String getb(){
String sql = "insert into ums_role_menu_relation value(32,33,33)";
jdbcTemplate.execute(sql);
fa();
return null;
}
//
@Transactional(propagation = Propagation.REQUIRED)
public void fa(){
String sql = "insert into ums_role_menu_relation value(33,33,33)";
jdbcTemplate.execute(sql);
throw new NullPointerException();
}
1.2 失效情况,
getb方法没有事务注解,但是有事务,此时,fa()会加入到getb()的事务中,但是fa()不是由代理对象调用,所以spring容器无法判断fa()方法上的注解,所以,当抛空指针后,也没有回滚
public String getb(){
String sql = "insert into ums_role_menu_relation value(32,33,33)";
jdbcTemplate.execute(sql);
fa();
return null;
}
//
@Transactional(propagation = Propagation.REQUIRED)
public void fa(){
String sql = "insert into ums_role_menu_relation value(33,33,33)";
jdbcTemplate.execute(sql);
throw new NullPointerException();
}
数据依然插入了
2、REQUIRES_NEW
(requires_new:需要新建):创建一个新事务,如果存在当前事务,则挂起该事务。
B会新建一个事务,A和B事务互不干扰,他们出现问题回滚的时候,也都只回滚自己的事务;
public String getb(){
String sql = "insert into ums_role_menu_relation value(32,33,33)";
jdbcTemplate.execute(sql);
serviceA.fa();
return null;
}
//
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void fa(){
String sql = "insert into ums_role_menu_relation value(33,33,33)";
jdbcTemplate.execute(sql);
throw new NullPointerException();
}