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

一、代购系统的定时任务痛点
一套完整的跨境代购业务流程包含:

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

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

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

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

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

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

@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的阻塞处理策略:

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

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

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中提到的“熔断+退避”机制:

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

Logo

电商企业物流数字化转型必备!快递鸟 API 接口,72 小时快速完成物流系统集成。全流程实战1V1指导,营造开放的API技术生态圈。

更多推荐