java - 跨微服务的事务
问题描述
我有两个实体:
实体 A:
private Long idA;
实体 B:
private Long idB;
private Long fkA;
我的微服务只能保存、修改或删除 B 实体:
存储库:
@Repository
public interface BRepository extends CrudRepository<B, Long>{
}
伪装外部服务:
@FeignClient(value = "aExtClient", url = "${A_EXT_MS_URL}")
public interface AExtRemoteService {
@DeleteMapping(path = "/a/{idA}", produces = MediaType.APPLICATION_JSON_VALUE)
public Esito deleteA(@PathVariable(name = "idA") Long idA);
}
服务实现:
@Component
@Transactional(rollbackFor = BusinessServiceException.class)
public class BServiceImpl implements BService {
@Autowired
private BRepository bRepository;
@Autowired
private AExtRemoteService aExtRemoteService;
@Override
public void deleteB(B b) {
Long idA = b.getFkA();
bRepository.delete(b); //marker
if(idA != null) {
aExtRemoteService.deleteA(idA);
}
}
}
外部服务只能保存、修改或删除 A 实体:
控制器:
@RestController
@RequestMapping("/a")
@CrossOrigin(origins = "${mocks.allowedCrossOrigins:*}")
public class AServiceMockController {
@Autowired
private AMockService aMockService;
@DeleteMapping(value = {
"/{idA}" }, produces = MediaType.APPLICATION_JSON_VALUE)
public @ResponseBody ResponseEntity<Result> deleteA(@PathVariable(name = "idA") Long idA) {
Result result = new Result();
aMockService.deleteA(idA);
result.setResult(true);
result.setCod(String.valueOf(HttpStatus.OK.value()));
result.setMess(HttpStatus.OK.name());
return new ResponseEntity<>(result, HttpStatus.OK);
}
}
服务:
@Service
public class AMockService {
private final ARepository aRepository;
@Autowired
public AMockService(ARepository aRepository) {
this.aRepository = aRepository;
}
public void deleteA(Long idA) {
A byId = aRepository.findById(idA).orElseThrow(NoSuchElementException);
aRepository.delete(byId);
}
}
存储库:
@Repository
public interface ARepository extends CrudRepository<A, Long>{
}
在调用 deleteB 时,外部服务已返回此消息:
Caused by: java.sql.SQLIntegrityConstraintViolationException: ORA-02292: integrity constraint violated (X.YY) - child record found
我该如何解决?似乎没有考虑到 bRepository.delete(b);
PS我可以访问外部服务,所以如果需要我可以修改它。
解决方案
您不能跨微服务进行此类事务 - 一个是因为您正在使用远程 http 调用服务 A 并且它本质上无法参与您在服务 B 中启动的事务。您正在使用的事务是资源本地事务。其次,您在服务中共享一个数据库,并且远程 http 服务正在尝试删除具有作为父级的外键关系的记录。由于您使用的是 Spring 事务管理器,因此您可以在 B 中的事务成功完成后使用TransactionEventListener(参见此处的示例用法)调用删除 A。还有其他模式可以解决跨微服务的事务,例如 Saga
推荐阅读
- css - 如何在角材料中嵌套网格列表?
- swift - NSManagedObject 的两个初始化器有什么区别
- python - 基于端点递归地划分列表
- python - 将字符串与数据框中的每一行连接起来并对其应用函数?
- php - 带有“DATEPART”和“DATEADD”的 SQL 语句不起作用。不被服务器识别
- asp.net-mvc - addTagHelper Html.GetEnumSelectList 不工作
- java - 带接口的 Getter 和 Setter 有一个参数
- javascript -
- php - Can't Upload folder to FTP Server : ftp_put(): SSL_shutdown failed
- sql - 如何在有条件的循环中选择列