工作上有个需求,A服务在调用B服务的方法之前需要先保存业务数据,B服务方法对业务数据进行计算,开启后续流程,然后再调用A服务的数据进行处理更新业务状态,整个流程都是连贯的。但是如果在最开始A服务的事务没有提交,后续的A服务通过ID查询业务数据的时候为空,但是整个系统都是有事务控制的,处理思路是在最开始的方法上加上注解 : @Transactional(propagation=Propagation.NOT_SUPPORTED) 取消事务,采用编程式事务控制;
@Autowired
private PlatformTransactionManager transactionManager;
public Serializable start(T flowAware) {
TransactionStatus status = transactionManager.getTransaction(new DefaultTransactionDefinition());
try {
if (flowAware.getDfId() == null) {
baseService.save(flowAware);
} else {
baseService.update(flowAware);
}
transactionManager.commit(status);
} catch (Exception e) {
transactionManager.rollback(status);
e.printStackTrace();
return null;
}
//调用接口,进行后续计算
try {
client.startProcess(flowAware);
} catch (Exception e) {
baseService.remove(flowAware);
e.printStackTrace();
}
return flowAware.getId();
}
编程式事务采用PlatformTransactionManager,通过commit提交事务,rollback回滚事务。
下面是事务的一些特性
@Transactional(propagation=Propagation.REQUIRED)//如果有事务,那么加入事务,没有的话新创建一个
@Transactional(propagation=Propagation.NOT_SUPPORTED)//这个方法不开启事务
@Transactional(propagation=Propagation.REQUIREDS_NEW)//不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务
@Transactional(propagation=Propagation.MANDATORY)//必须在一个已有的事务中执行,否则抛出异常
@Transactional(propagation=Propagation.NEVER)//不能在一个事务中执行,就是当前必须没有事务,否则抛出异常
@Transactional(propagation=Propagation.SUPPORTS)//其他bean调用这个方法,如果在其他bean中声明了事务,就是用事务。没有声明,就不用事务。
@Transactional(propagation=Propagation.NESTED)//如果一个活动的事务存在,则运行在一个嵌套的事务中,如果没有活动的事务,则按照REQUIRED属性执行,它使用一个单独的事务。这个书屋拥有多个回滚的保存点,内部事务的回滚不会对外部事务造成影响,它只对DataSource TransactionManager事务管理器起效。
@Transactional(propagation=Propagation.REQUIRED,readOnly=true)//只读,不能更新,删除
@Transactional(propagation=Propagation.REQUIRED,timeout=30)//超时30秒
@Transactional(propagation=Propagation.REQUIRED,isolation=Isolation.DEFAULT)//数据库隔离级别