Spring Boot + XXL-JOB 搭建淘宝代购系统任务调度中心

最近在给一家代购系统服务商做技术顾问,他们的业务是帮海外华人代购国内商品,每天几万单,需要定时抓取淘宝/1688的物流轨迹。之前用的是linux cron,管理混乱。我帮他们引入了XXL-JOB,并结合taocarts的任务设计思路重写了调度层。

一、代购系统的定时任务痛点

一套完整的跨境代购业务流程包含:

每10分钟抓取用户收藏的商品价格变动(反向代购场景下价格敏感)

每30分钟同步1688订单的卖家发货状态

每天凌晨计算代购集运包裹的仓储费

不定期执行国际集运路线的运费试算

如果任务之间没有依赖管理和失败重试,一个任务阻塞会拖垮整个系统。而taocarts项目里用了分布式任务调度,我参考它的表结构,最终选定XXL-JOB。

二、集成XXL-JOB与taocarts的适配思路

taocarts 的任务模块是把所有Job定义为interface,然后通过反射调用。我们直接用XXL-JOB的Bean模式,把每个Job写成一个Spring Bean:

java 复制代码
@Component
@JobHandler(value = "syncTaobaoOrderStatusJob")
public class SyncTaobaoOrderStatusJob extends IJobHandler {
    
    @Autowired
    private OrderService orderService;
    @Autowired
    private TaobaoApiClient taobaoClient;

    @Override
    public ReturnT<String> execute(String param) throws Exception {
        // 查询最近1小时内待同步的订单
        List<Order> orders = orderService.listPendingSync(LocalDateTime.now().minusHours(1));
        for (Order order : orders) {
            try {
                String status = taobaoClient.getOrderStatus(order.getPlatformOrderId());
                order.setPlatformStatus(status);
                orderService.updateById(order);
                // 如果订单已签收,触发**代购转运**完成通知
                if ("SIGNED".equals(status)) {
                    eventPublisher.publishOrderSignedEvent(order);
                }
            } catch (Exception e) {
                XxlJobLogger.log("同步订单失败, orderId: {}", order.getId());
                // 记录失败,下次重试
                failRecorder.record(order.getId());
            }
        }
        return SUCCESS;
    }
}

为了让任务能动态配置淘宝1688代购系统的店铺分组,我们扩展了XXL-JOB的阻塞处理策略:

java 复制代码
// 自定义阻塞策略:当任务堆积时,跳过本次执行,但保留最后调度时间
if (executorBlockStrategy == BlockStrategyEnum.DISCARD_LATER) {
    long lastExecTime = jobThread.getLastExecTime();
    if (System.currentTimeMillis() - lastExecTime < 5000) {
        XxlJobLogger.log("任务执行间隔过短,直接丢弃");
        return new ReturnT<>(ReturnT.FAIL_CODE, "丢弃");
    }
}

三、结合代购转运的物流轨迹抓取

代购转运环节中,我们需要抓取国内快递(中通、圆通)的轨迹,合并到国际段。原来代码中每个快递公司一个独立Job,维护成本高。参考taocarts的"策略工厂"模式,我们改成了统一接口:

java 复制代码
public interface LogisticsTracker {
    TrackingInfo track(String trackingNumber);
}

@Component
public class LogisticsFactory {
    private Map<String, LogisticsTracker> trackerMap;
    
    public LogisticsTracker getTracker(String companyCode) {
        return trackerMap.getOrDefault(companyCode, new DefaultTracker());
    }
}

然后在XXL-JOB中只用一个Job,传入参数"company=zt&no=123456",根据公司码选择具体实现。这大大简化了代购系统的维护。

四、生产环境的一个大坑:1688接口限流导致任务雪崩

双十一期间,1688的开放平台限流变严格,我们的订单同步任务大量超时,而XXL-JOB默认会重试3次,导致调度中心队列爆满。最后用了taocarts中提到的"熔断+退避"机制:

java 复制代码
@XxlJob("sync1688Order")
public void sync1688Order() {
    if (circuitBreaker.isOpen()) {
        XxlJobLogger.log("熔断器打开,本次跳过");
        return;
    }
    try {
        doSync();
        circuitBreaker.recordSuccess();
    } catch (RateLimitException e) {
        circuitBreaker.recordFailure();
        // 退避等待
        Thread.sleep(circuitBreaker.getRetryDelayMs());
    }
}

这个改进让我们平稳度过了高峰期。如果你也在开发代购源码,强烈建议加上熔断。

五、总结

任务调度是跨境独立站和代购系统最容易被忽视的基础设施。taocarts 给了我很多灵感,但生产级系统必须考虑降级和限流。XXL-JOB 简单可靠,搭配好阻塞策略,足够支撑日活10万级的反向海淘平台。

相关推荐
KobeSacre10 分钟前
ReentrantLock源码
java
嵌入式协会202407215 分钟前
(已解决)MinIO python 获取预签名出现forbidden、errornetwork等错误
java·开发语言·python
不才不才不不才38 分钟前
Spring AI 实战:聊天、提示词、记忆三件套
java·人工智能·spring·ai
一 乐2 小时前
家政服务管理系统|基于springboot + vue家政服务管理系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·家政服务管理系统
IT_陈寒2 小时前
Vite热更新失效?可能你在用Windows
前端·人工智能·后端
碳基硅坊3 小时前
Spring AI:把大模型接进 Spring 应用
java·人工智能·spring ai
椰椰椰耶3 小时前
[SpringCloud][14]OpenFeign参数传递方法
后端·spring·spring cloud
黄毛火烧雪下3 小时前
Java 核心知识点总结(一)
java·开发语言
onething3654 小时前
Spring Boot + Spring AI 从入门到实战:7天转型计划 Day 3 —— 消息表设计 + 级联删除 + 事务管理
人工智能·后端
荣江4 小时前
Hermes Agent 代码仓库打包工具使用指南(repomix-rs 高性能版)
后端