【数模电路】NE555定时器超详细底层原理
2026/6/26 5:58:20
如何保证“本地数据库插入”与“调用第三方接口”这两个操作的原子性(要么都成功,要么都失败),这是一个非常经典且常见的分布式事务场景。
这是最简单且最推荐的方案,核心思想是先落库,再同步。
1.逻辑设计:
2.异常处理(补偿机制):
示例代码:
// 1. 本地保存(事务内) @Transactional(rollbackFor = Exception.class) public void saveLocal(Data data) { data.setSyncStatus("UN_SYNC"); mapper.insert(data); } // 2. 主业务逻辑 public void saveAndSync(Data data) { // 第一步:先落库(保证本地数据安全) saveLocal(data); // 第二步:尝试同步(即使这里失败了,也不会回滚上面的saveLocal) try { boolean result = thirdPartyClient.call(data); if (result) { // 同步成功,更新状态 mapper.updateStatus(data.getId(), "SYNCED"); } } catch (Exception e) { log.error("同步失败,等待定时任务补偿", e); // 这里吞掉异常,不要影响主流程返回成功 } } // 3. 另外编写一个定时任务 (例如每5分钟执行一次) @Scheduled(cron = "0 0/5 * * * ?") public void retrySyncTask() { List<Data> pendingData = mapper.selectUnSyncData(); for (Data data : pendingData) { // 重新调用第三方,成功后更新状态 // 建议设置最大重试次数,超过后人工介入 } }如果您的系统已经引入了消息队列(如 RabbitMQ, RocketMQ, Kafka),可以使用可靠消息服务。
1.逻辑设计:
2.优点: 解耦了业务逻辑和第三方调用,系统吞吐量高。
3.注意: 消费者端需要做好幂等性处理(防止第三方接口被重复调用)。
这里是关键:业务数据入库和本地消息入库必须在同一个 @Transactional 事务中。
@Service public class UserService { @Autowired private UserMapper userMapper; @Autowired private LocalMessageMapper messageMapper; @Autowired private RabbitMQService rabbitMQService; // 封装的MQ发送服务 /** * 注册用户并触发同步 */ @Transactional(rollbackFor = Exception.class) // 开启本地事务 public void registerUser(User user) { // 1. 插入业务数据 userMapper.insert(user); // 2. 组装消息内容 String msgId = UUID.randomUUID().toString(); UserSyncMsg msgContent = new UserSyncMsg(user.getId(), user.getName()); String json = JSON.toJSONString(msgContent); // 3. 插入本地消息表 (状态为 0-待发送) // 这步非常关键!它保证了如果数据库回滚,消息记录也会回滚;如果提交,消息记录一定存在。 LocalMessage localMsg = new LocalMessage(); localMsg.setId(msgId); localMsg.setMsgContent(json); localMsg.setExchange("user.exchange"); localMsg.setRoutingKey("user.sync.crm"); localMsg.setStatus(0); messageMapper.insert(localMsg); // 4. 发送MQ消息 (这步其实可以异步,或者放在事务提交后的回调中,这里为了简单直接调用) // 注意:即使这里发MQ失败报错,也不要抛出异常导致事务回滚。 // 因为我们有定时任务兜底(见第4步)。 try { rabbitMQService.send(msgId, json); // 发送成功,更新本地消息状态为 1-已发送 messageMapper.updateStatus(msgId, 1); } catch (Exception e) { log.error("MQ发送失败,等待定时任务补偿: {}", msgId); // 这里吞掉异常,不要影响主业务注册成功 } } }消费者负责调用第三方接口。如果失败,利用 MQ 的重试机制或死信队列。
@Component @RabbitListener(queues = "user.sync.queue") public class UserSyncConsumer { @Autowired private ThirdPartyCrmClient crmClient; @RabbitHandler public void process(String msgJson, Channel channel, @Header(AmqpHeaders.DELIVERY_TAG) long tag) { UserSyncMsg msg = JSON.parseObject(msgJson, UserSyncMsg.class); try { // 1. 幂等性检查 (非常重要!) // 调用三方接口前,先查一下三方或者本地Redis,确保这个userId没有被同步过。 // if (isSynced(msg.getUserId())) { channel.basicAck(tag, false); return; } // 2. 调用第三方接口 boolean success = crmClient.syncUserToCrm(msg); if (success) { // 3. 成功,手动确认消息 (ACK) channel.basicAck(tag, false); log.info("同步第三方成功: {}", msg.getUserId()); } else { // 4. 业务逻辑失败(比如参数校验不过),通常不再重试,或者进入死信队列人工处理 log.error("第三方返回失败"); channel.basicNack(tag, false, false); // 不重回队列,转入死信或丢弃 } } catch (Exception e) { // 5. 网络抖动等异常,拒绝消息并重回队列 (Requeue = true) // 这样MQ过一会会再次推送这条消息 try { // 也可以结合重试次数判断,如果重试太多次就丢进死信队列 channel.basicNack(tag, false, true); } catch (IOException ex) { ex.printStackTrace(); } } } }这是“最终一致性”的保障。防止第2步中 rabbitMQService.send 失败(例如MQ挂了),导致本地消息表一直是“待发送”状态。
@Component public class MessageResendTask { @Autowired private LocalMessageMapper messageMapper; @Autowired private RabbitMQService rabbitMQService; // 每分钟扫描一次状态为 0 (待发送) 且创建时间超过1分钟的消息 @Scheduled(fixedRate = 60000) public void resendFailedMessages() { List<LocalMessage> failedMsgs = messageMapper.selectPendingMessages(); for (LocalMessage msg : failedMsgs) { if (msg.getRetryCount() > 5) { // 超过最大重试次数,标记为失败,报警人工介入 messageMapper.updateStatus(msg.getId(), 2); continue; } try { rabbitMQService.send(msg.getId(), msg.getMsgContent()); // 发送成功,更新状态 messageMapper.updateStatus(msg.getId(), 1); } catch (Exception e) { // 再次失败,增加重试次数 messageMapper.incrementRetryCount(msg.getId()); } } } }这个 Demo 实现了以下逻辑闭环:
如果您不想引入复杂的 MQ 或定时任务,可以在代码层面做简单的“重试”。
1.逻辑设计:
2.重大缺陷:
如果业务场景非常严格,要求强一致性(几乎同时成功或失败),可以使用分布式事务框架(如 Alibaba Seata)。
综合考虑开发成本和系统稳定性,方案一(本地消息表+定时任务补偿) 是性价比最高的选择。