产生分布式事务的最大原因:网络抖动, A服务(插入数据,回滚了)----->B服务不知(更新数据成功)
本地事务注解: @Transactional使用
1. @Transactional 注解只能用在public 方法上,如果用在protected或者private的方法上,不会报错,但是该注解不会生效。
2. @Transactional 注解默认的 rollbackFor 是运行时异常,具体为RuntimeException及其子类.
3.同一个类即同一个Service的成员事务方法互调默认是不生效的,因为绕过了代理对象,比如:
@Slf4j
@Service("test")
public class UserImpl extends ServiceImpl<UserMapper, UserEntity> implements UserService {
@Transactional
public void a(){
this.b();
this.c();
}
//无效
@Transactional
public void b(){
//TODO
}
//无效
@Transactional
public void c(){
//TODO
}
}
@Slf4j
@Service("test")
public class UserImpl extends ServiceImpl<UserMapper, UserEntity> implements UserService {
@Transactional
public void a(){
B.b();
C.c();
}
需要改成________________________________________________
@Service("B")
public class BImpl{
@Transactional
public void b(){
//TODO
}
}
@Service("C")
public class BImpl{
@Transactional
public void b(){
//TODO
}
}
方案2:
<!--AOP 解决本类的本地事务(方法)互调控制失效问题-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
启动类添加 @EnableAspectJAutoProxy(exposeProxy = true) //开启aspectj动态代理
以后动态代理都是由aspectj创建(没有公共接口也可以创建动态代理),不再是默认的JDK创建.
@Slf4j
@Service("test")
public class UserImpl extends ServiceImpl<UserMapper, UserEntity> implements UserService {
@Transactional
public void a(){
UserImpl userImpl = (UserImpl) AopContext.currentProxy();
userImpl.b();
userImpl.c();
}
@Transactional
public void b(){
//TODO
}
@Transactional
public void c(){
//TODO
}
}