导购app佣金模式微服务拆分:领域驱动设计在返利系统中的实践

导购app佣金模式微服务拆分:领域驱动设计在返利系统中的实践

大家好,我是省赚客APP研发者阿宝!"省赚客"作为聚娃科技旗下的导购返利平台,其核心业务围绕用户推广、订单跟踪与佣金结算展开。早期单体架构下,佣金计算逻辑与订单、商品、用户模块高度耦合,导致迭代缓慢、测试困难。为提升系统可维护性与扩展性,我们采用领域驱动设计(DDD)方法论,对返利系统进行微服务拆分,明确限界上下文(Bounded Context),并以聚合根保障业务一致性。

识别核心子域与限界上下文

通过事件风暴工作坊,我们识别出三个核心子域:

  • 推广域(Promotion):管理用户生成的推广链接、任务配置
  • 订单域(Order):处理电商平台回传的订单数据
  • 佣金域(Commission):负责佣金计算、状态机与结算

每个子域对应一个独立微服务,拥有专属数据库与API契约。

佣金聚合根设计

佣金域的核心是Commission聚合根,它封装了佣金生命周期的所有规则:

java 复制代码
package juwatech.cn.commission.domain;

import java.math.BigDecimal;
import java.time.LocalDateTime;

public class Commission {
    private String id;
    private String userId;
    private String orderId;
    private BigDecimal amount;
    private CommissionStatus status;
    private LocalDateTime confirmedAt;
    private LocalDateTime settledAt;

    public void confirm(BigDecimal actualAmount) {
        if (this.status != CommissionStatus.PENDING) {
            throw new IllegalStateException("Only PENDING commission can be confirmed");
        }
        this.amount = actualAmount;
        this.status = CommissionStatus.CONFIRMED;
        this.confirmedAt = LocalDateTime.now();
    }

    public void settle() {
        if (this.status != CommissionStatus.CONFIRMED) {
            throw new IllegalStateException("Only CONFIRMED commission can be settled");
        }
        this.status = CommissionStatus.SETTLED;
        this.settledAt = LocalDateTime.now();
    }

    // getters omitted
}

该聚合根确保状态变更符合业务规则,防止非法跃迁。

领域事件驱动跨服务协作

当订单域确认一笔有效订单后,发布OrderConfirmedEvent,佣金服务监听并创建佣金记录:

java 复制代码
// 订单服务发布事件
package juwatech.cn.order.event;

public class OrderConfirmedEvent {
    private String orderId;
    private String userId;
    private String itemId;
    private BigDecimal orderAmount;
    private LocalDateTime confirmedTime;
    // constructor, getters
}

// 佣金服务消费事件
package juwatech.cn.commission.application;

import org.springframework.kafka.annotation.KafkaListener;
import org.springframework.stereotype.Service;

@Service
public class CommissionEventHandler {

    private final CommissionService commissionService;

    public CommissionEventHandler(CommissionService commissionService) {
        this.commissionService = commissionService;
    }

    @KafkaListener(topics = "order-confirmed")
    public void handleOrderConfirmed(OrderConfirmedEvent event) {
        // 查询推广关系,确定是否应计佣
        PromotionRelation relation = promotionClient.getRelation(event.getUserId(), event.getItemId());
        if (relation != null) {
            BigDecimal rate = taskConfigClient.getRate(relation.getTaskId());
            BigDecimal commissionAmount = event.getOrderAmount().multiply(rate);
            commissionService.createPendingCommission(
                event.getOrderId(),
                event.getUserId(),
                commissionAmount
            );
        }
    }
}

该机制解耦订单与佣金服务,避免直接调用。

防腐层(Anti-Corruption Layer)隔离外部依赖

佣金服务需调用推广服务获取任务配置,但不希望暴露其内部模型。我们通过防腐层转换:

java 复制代码
package juwatech.cn.commission.infra.ac;

import juwatech.cn.commission.domain.TaskRate;
import juwatech.cn.promotion.api.dto.TaskConfigDto;

public class PromotionAcLayer {
    private final PromotionFeignClient client;

    public PromotionAcLayer(PromotionFeignClient client) {
        this.client = client;
    }

    public TaskRate getTaskRate(String taskId) {
        TaskConfigDto dto = client.getTaskConfig(taskId);
        return new TaskRate(dto.getId(), new BigDecimal(dto.getCommissionRate()));
    }
}

TaskRate是佣金域自有模型,不受推广服务变更影响。

Saga模式保障跨服务最终一致性

佣金创建、用户余额更新、账单生成涉及多个服务,我们采用Saga模式实现补偿:

java 复制代码
package juwatech.cn.commission.application;

import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service
public class CommissionSettlementSaga {

    @Transactional
    public void executeSettlement(String commissionId) {
        try {
            commissionDomain.confirm(commissionId); // 1. 确认佣金
            balanceClient.increaseBalance(userId, amount); // 2. 增加余额
            billClient.createBill(commissionId, amount); // 3. 生成账单
        } catch (Exception e) {
            // 触发补偿
            compensationService.compensate(commissionId, e);
        }
    }
}

补偿服务按逆序执行:

java 复制代码
public void compensate(String commissionId, Exception cause) {
    if (billExists(commissionId)) {
        billClient.cancelBill(commissionId);
    }
    if (balanceIncreased(commissionId)) {
        balanceClient.decreaseBalance(userId, amount);
    }
    commissionDomain.rollbackToPending(commissionId);
}

虽非强一致,但在金融容忍范围内满足业务需求。

CQRS优化查询性能

佣金列表页需展示状态、金额、订单信息等,若从聚合根加载效率低下。我们采用CQRS模式,写模型更新后同步写入读模型:

java 复制代码
// 写操作完成后发布事件
applicationEventPublisher.publishEvent(new CommissionUpdatedEvent(commissionId));

// 读模型监听器
@EventListener
public void onCommissionUpdated(CommissionUpdatedEvent event) {
    Commission commission = commissionRepository.findById(event.getId());
    CommissionView view = viewMapper.toView(commission);
    viewRepository.save(view); // 存入MySQL读库
}

前端直接查询commission_view表,避免复杂JOIN。

本文著作权归聚娃科技省赚客app开发者团队,转载请注明出处!

相关推荐
独特的螺狮粉7 小时前
城市空气质量简易指数查询卡片:鸿蒙Flutter框架 实现的空气质量查询应用
开发语言·flutter·华为·架构·harmonyos
架构师老Y8 小时前
011、消息队列应用:RabbitMQ、Kafka与Celery
python·架构·kafka·rabbitmq·ruby
沃尔威武8 小时前
微服务架构下:如何用gRPC实现跨语言高效通信
微服务·云原生·架构
Rick19938 小时前
LangChain 核心解析:底层架构、原理
架构·langchain
heimeiyingwang8 小时前
【架构实战】数据加密架构:传输加密+存储加密
架构
2501_948114249 小时前
Claude Sonnet 4.6 深度评测:性能逼近 Opus、成本打骨折,附接入方案与选型指南
大数据·网络·人工智能·安全·架构
张3239 小时前
K8s 标签
云原生·容器·kubernetes
indexsunny9 小时前
互联网大厂Java面试实战:从Spring Boot到微服务架构的深度探讨
java·数据库·spring boot·安全·微服务·监控·面试实战
Meme Buoy9 小时前
17.补充:知识产权和标准化
架构
两万五千个小时9 小时前
Claude Code 源码:工具 Plan 模式
人工智能·程序员·架构