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万级的反向海淘平台。

相关推荐
一行代码一行诗++2 小时前
goto语句
java·开发语言·算法
m0_639310793 小时前
大数据技术原理-HDFS的安装与应用
java·大数据·jvm·hadoop·spring·hdfs·eclipse
kunge20133 小时前
Claude Code 工作流中的命令实现与自定义指南
人工智能·后端·架构
明月_清风3 小时前
Go 没有 `class`,如何实现面向对象三要素?与传统 OOP 的深度对比
后端·go
Plastic garden3 小时前
Redis(2) redis的高可用
java·数据库·redis
XiYang-DING3 小时前
【Spring】SpringIoC&DI
java·spring·log4j
xixingzhe23 小时前
spring构造函数注入对比@Resource
java·后端·spring
程序员牛奶3 小时前
[Algo-2]双指针技巧:你真的学懂双指针了吗?
后端
宋哥转AI3 小时前
Java搭RAG实战(三):检索问答全链路,从架构分层到SSE流式
java·agent