支付回调幂等性处理:Redis分布式锁 + 数据库唯一键

简介: 本文提出三层幂等方案应对跨境电商支付回调重复问题:Redis分布式锁(防并发)、数据库唯一索引(防插入重复)、订单状态前置校验(防业务重复)。Taoify系统应用后实现零重复订单,日均万级回调准确处理,并辅以定时补偿确保最终一致性。(239字)

摘要:支付回调重复通知是跨境电商系统的常见问题,处理不当会导致重复发货、重复扣款。本文给出三层幂等保障方案:Redis分布式锁、数据库唯一索引、订单状态前置检查。Taoify跨境电商通过该方案实现了零重复订单,日均处理数万次回调无误。
一、问题场景
PayPal、Stripe等支付网关为保证送达,会多次发送同一笔交易的回调。如果系统未做幂等处理,可能重复更新订单状态,触发多次发货、发送多封邮件,甚至重复扣减库存。
二、解决方案架构
第一层:Redis分布式锁。以支付流水号(transaction_id)为key,设置锁超时30秒。同一流水号的回调只有获取锁的线程才能处理。
java
@Servicepublic class PaypalWebhookService { @Autowired private RedisTemplate redisTemplate; public void handleWebhook(PaypalEvent event) { String transactionId = event.getResource().getId(); String lockKey = "paypal:lock:" + transactionId; Boolean locked = redisTemplate.opsForValue().setIfAbsent(lockKey, "1", Duration.ofSeconds(30)); if (Boolean.FALSE.equals(locked)) { log.info("重复回调,已忽略: {}", transactionId); return; } try { doHandle(event); } finally { redisTemplate.delete(lockKey); } }}
第二层:数据库唯一键。支付回调记录表对transaction_id建立唯一索引,保证同一支付流水号只能插入一次。
sql
CREATE TABLE payment_callback_log ( id BIGINT PRIMARY KEY AUTO_INCREMENT, transaction_id VARCHAR(64) NOT NULL COMMENT '支付流水号', channel VARCHAR(20) NOT NULL, payload TEXT, created_at DATETIME, UNIQUE KEY uk_transaction_id (transaction_id));
java
@Transactionalpublic void doHandle(PaypalEvent event) { try { paymentCallbackLogMapper.insert(log); // 可能因唯一键冲突抛异常 // 更新订单状态 orderService.updateStatus(event.getOrderNo(), OrderStatus.PAID); } catch (DuplicateKeyException e) { log.info("回调已处理过: {}", event.getTransactionId()); return; }}
第三层:订单状态前置检查。在更新订单前,再次查询当前订单状态,若已经是“已支付”或“待采购”,则直接返回。
java
public void updateOrderStatus(String orderNo, OrderStatus targetStatus) { Order order = orderMapper.selectByOrderNo(orderNo); if (order.getStatus() == OrderStatus.PAID || order.getStatus() == OrderStatus.PENDING_PURCHASE) { log.info("订单状态已更新,无需重复处理: {}", orderNo); return; } // 乐观锁更新 int rows = orderMapper.updateStatus(orderNo, oldStatus, targetStatus); if (rows == 0) { throw new OptimisticLockException("订单状态已被其他线程修改"); }}
三、补偿机制
极端情况下,回调可能丢失(网络问题)。Taoify跨境电商还实现了定时任务,扫描“待支付”状态超过30分钟的订单,主动向支付网关查询支付状态,确保最终一致性。
java
@Scheduled(cron = "0 /10 ?")public void syncPaymentStatus() { List pendingOrders = orderMapper.selectByStatus(OrderStatus.PENDING_PAYMENT, Duration.ofMinutes(30)); for (Order order : pendingOrders) { PaymentStatus status = paymentGateway.query(order.getTransactionId()); if (status == PaymentStatus.SUCCESS) { orderService.updateStatus(order.getOrderNo(), OrderStatus.PAID); } else if (status == PaymentStatus.FAILED) { orderService.updateStatus(order.getOrderNo(), OrderStatus.CANCELLED); } }}

目录
相关文章
|
18天前
|
人工智能 自然语言处理 文字识别
阿里云百炼Qwen3.7-Max简介:能力、优势、支持订阅计划参考
Qwen3.7-Max是阿里云百炼面向智能体时代推出的新一代旗舰模型,对标GPT-5.5、Claude Opus 4.7等闭源旗舰。该模型支持百万级token上下文窗口,具备顶级推理能力、多模态搜索与视觉理解增强、流式输出低延迟响应等核心优势,覆盖编程、办公、长周期自主执行等复杂场景。同时支持OpenAI接口兼容,便于系统快速迁移。用户可通过Token Plan团队或节省计划等订阅方式灵活调用,适合企业级高要求场景使用。
6797 30
阿里云百炼Qwen3.7-Max简介:能力、优势、支持订阅计划参考
|
3天前
|
数据采集 人工智能 前端开发
让 Coding Agent 从黑盒到透明:阿里云 Agent 观测审计数据采集实践
AI Agent 规模化落地带来执行黑盒、行为难追溯、成本难度量三大难题。阿里云基于 OTel 标准,面向 Coding Agent、个人通用助理和框架型 Agent,推出 LoongSuite Pilot、插件及探针等无侵入采集方案,让 Agent 实现可看见、可分析、可审计、可治理。
605 138
|
3天前
|
人工智能 弹性计算 运维
阿里云发布堡垒机智能运维Agent,运维交互进入自然语言新时代
支持自然语言运维,提升效率与安全双保障。
1145 0
|
10天前
|
人工智能 安全 定位技术
CodeGraph深度解析 让Claude Code工具调用直降七成的核心原理与实操教程
如今以Claude Code为代表的AI编程智能体已经成为开发者日常编码、项目重构、漏洞修复的必备工具。但在长期使用过程中,几乎所有开发者都会遇到同一个明显痛点:AI虽然具备强大的代码生成与分析能力,却常常陷入盲目探索的循环中。
1164 1
|
13天前
|
存储 定位技术 数据库
CodeGraph 如何让 Claude Code减少 7 成工具调用?
CodeGraph 为 Coding Agent 提供本地代码知识图谱,把函数、类、调用链和框架路由提前整理成“项目地图”,减少盲目搜索和文件读取。它不是新 Agent,而是上下文基础设施,让 Agent 更快找到正确代码路径,平均减少 7 成工具调用。
1270 3
|
11天前
|
人工智能 弹性计算 安全
阿里云618活动时间、活动入口、优惠活动详细解读
2026年阿里云618创新加速季已全面开启,作为年度力度最大的云产品促销活动,本次大促覆盖轻量应用服务器、ECS云服务器、GPU云服务器、数据库、AI算力、安全服务、CDN等全品类产品,推出5亿元算力补贴、新用户限时秒杀、普惠满减、企业专享、免费试用、云大使返佣等多重福利,个人开发者、中小企业、AI团队均可享受专属低价。本文将系统梳理2026年阿里云618活动的完整时间节点、官方参与入口、各类优惠细则、使用规则、热门产品推荐及实操代码,帮助用户精准参与、高效省钱,以最低成本完成上云部署。
965 5
|
9天前
|
人工智能 自然语言处理 安全
Vibe Coding 实战:别盲目跟风,先分清 vibe coding 适合什么场景
本文系统总结vibe coding实战经验:明确其适用场景(原型、小工具、标准化模块),剖析5步落地流程(场景判定→结构化提示词→目录初始化→分模块生成→自动化校验),指出四大常见误区,并推荐适配工具Trae。强调“场景匹配+规则前置”是提效关键,避免盲目套用。
798 1