一次DDD重构实践

我来问道无馀说,云在青天水在瓶。
--- 唐.李翱《赠药山高僧惟俨二首⑴ 》

最近在开发一个配额创建和修改的功能,需求很简单:

  1. 配额能够正确创建,比如:2019-05-25, 用户lxy创建了一个配额,包含10件某商品。订单的状态包括:新建、提交等等。

  2. 每个配额单能够分配到具体的商店。
    比如第一步创建的订单,能够分配到两个商店,A商店4件,B商店6件。

  3. 配额单创建后能够修改。
    3-1 配额商品数量能够修改,比如由10件变成15件。
    3-2 配额商品分配到商店的信息能够添加或修改,比如分配到A商店的4件商品变成6件,增加一个条目,分配3件商品到C商店。

数据库表是这样设计的(省去审计字段),

quota表:

id product_id quntity status
quotaid productid 100 0

quota_item表:

id quota_id shop_id quntity
item_id quotaid shopid 20
item_id2 quotaid shopid2 80

需求比较清晰,和BA、QA沟通理解一致后,开始开发。

项目用springboot框架开发,使用洋葱架构分层,核心代码如下:

Controller:
@RestController
public class QuotaController implements QuotaApi {
    private final QuotaApplicationService applicationService;

    public QuotaController(QuotaApplicationService applicationService) {
        this.applicationService = applicationService;
    }

    @Override
    public QuotaCreateResponse createQuota(@RequestBody QuotaCreateRequest request) {
        return applicationService.createQuota(
                Quota.builder()
                        .productId(request.getProductId())
                        .quantity(request.getQuantity()).status("create").build());
    }

    @Override
    public QuotaItemCreateResponse createQuotaItem(@RequestBody QuotaItemCreateRequest request) {
        QuotaItemCreateCommand command = QuotaItemMapper.MAPPER.toCommand(request);
        return applicationService.createQuotaItem(command);
    }
}

ApplicationService:
@Service
public class QuotaApplicationService {
    private final QuotaRepository repository;
    private final QuotaItemRepository itemRepository;

    public QuotaApplicationService(QuotaRepository repository, QuotaItemRepository itemRepository) {
        this.repository = repository;
        this.itemRepository = itemRepository;
    }

    public QuotaCreateResponse createQuota(Quota quota) {
        Quota result = repository.createQuota(quota);

        return QuotaMapper.MAPPER.toResponse(result);
    }

    public QuotaItemCreateResponse createQuotaItem(QuotaItemCreateCommand command) {
        List<String> shopIds = command.getQuotaItemDtoList().stream()
                .map(QuotaItemCreateCommand.QuotaItemDto::getShopId)
                .collect(Collectors.toList());

        List<QuotaItem> existQuotaItems = searchQuotaItems(command.getQuotaId(), shopIds);
        QuotaHelper helper = new QuotaHelper(command, existQuotaItems);

        Quota quota = repository.updateQuota(helper.getUpdateQuota());
        List<QuotaItem> insertResult = itemRepository.saveAll(helper.getInsertQuotaItems());
        List<QuotaItem> updateResult = itemRepository.updateAll(helper.getUpdateQuotaItems());

        return buildResponse(quota, insertResult, updateResult);
    }

    private List<QuotaItem> searchQuotaItems(String quotaId, List<String> shopIds) {
        return repository.queryQuotaItems(quotaId, shopIds);
    }

    private static class QuotaHelper {
        private final QuotaItemCreateCommand command;
        private final List<QuotaItem> existQuotaItems;

        QuotaHelper(QuotaItemCreateCommand command, List<QuotaItem> existQuotaItems) {
            this.command = command;
            this.existQuotaItems = existQuotaItems;
        }

        Quota getUpdateQuota() {
            return Quota.builder().id(command.getQuotaId()).quantity(command.getQuantity()).status("update").build();
        }

        List<QuotaItem> getInsertQuotaItems() {
            List<QuotaItemCreateCommand.QuotaItemDto> createDtos = command.getQuotaItemDtoList().stream()
                    .filter(quotaItemDto -> existQuotaItems.stream()
                            .noneMatch(quotaItem -> quotaItemDto.getShopId().equals(quotaItem.getShopId())))
                    .collect(Collectors.toList());

            return QuotaItemMapper.MAPPER.toDomain(createDtos).stream().peek(quotaItem -> {
                quotaItem.setId(UUID.randomUUID().toString());
                quotaItem.setQuotaId(command.getQuotaId());
            }).collect(Collectors.toList());
        }

        List<QuotaItem> getUpdateQuotaItems() {
            existQuotaItems.forEach(quotaItem -> {
                Optional<QuotaItemCreateCommand.QuotaItemDto> updateItem =
                        command.getQuotaItemDtoList().stream()
                                .filter(quotaItemDto -> quotaItemDto.getShopId().equals(quotaItem.getShopId()))
                                .findFirst();

                updateItem.ifPresent(quotaItemDto -> quotaItem.setQuantity(quotaItemDto.getQuantity()));
            });

            return existQuotaItems;
        }
    }

    private QuotaItemCreateResponse buildResponse(Quota quota, List<QuotaItem> insertResult, List<QuotaItem> updateResult) {
        updateResult.addAll(insertResult);

        return QuotaItemCreateResponse.builder()
                .quotaId(quota.getId())
                .quantity(quota.getQuantity())
                .quotaItemDtoList(QuotaItemMapper.MAPPER.toResponse(updateResult))
                .build();
    }
}

Domain:
public class Quota {
    private String id;
    private String productId;
    private Long quantity;
    private String status;
}

public class QuotaItem {
    private String id;
    private String quotaId;
    private String shopId;
    private Long quantity;
}

代码实现很简单,主要逻辑在QuotaApplicationService里,在创建QuotaItem时引入了一个辅助类QuotaHelper,判断哪些Item是需要更新的,哪些是新创建的,然后分别做更新和创建。至于domain,完全是贫血模型,没有承担任何业务逻辑,类型于传统的JavaBean

团队code diff后,小伙伴们指出了代码中存在的明显缺陷。提炼如下:

在数据库层面,quota和quotaItem是两张表,但是在领域模型层面。quotaItem是从属于quota的,quota没有quotaItem是可以独立存在的。但是,quotaItem必须属于一个quota。

按照这个思路,对代码进行重构,主要是对领域模型的重构以及因此带来的变更

domain:
public class Quota {
    private String id;
    private String productId;
    private Long quantity;
    private String status;

    private List<QuotaItem> items;

    public void update(Long quantity, List<QuotaItemCreateCommand.QuotaItemDto> quotaItemDtoList) {
        this.quantity = quantity;
        Map<String, QuotaItem> itemDtoMap =
                items.stream().collect(Collectors.toMap(QuotaItem::getShopId, Function.identity()));

        this.items = quotaItemDtoList.stream().map(quotaItemDto -> {
            if (itemDtoMap.containsKey(quotaItemDto.getShopId())) {
                QuotaItem quotaItem = itemDtoMap.get(quotaItemDto.getShopId());
                quotaItem.setQuantity(quotaItemDto.getQuantity());

                return quotaItem;
            }

            return QuotaItem.builder()
                    .id(UUID.randomUUID().toString())
                    .quotaId(this.getId())
                    .shopId(quotaItemDto.getShopId())
                    .quantity(quotaItemDto.getQuantity())
                    .build();
        }).collect(Collectors.toList());
    }
}

controller:
@RestController
public class QuotaController implements QuotaApi {
    private final QuotaApplicationService applicationService;

    public QuotaController(QuotaApplicationService applicationService) {
        this.applicationService = applicationService;
    }

    @Override
    public QuotaCreateResponse createQuota(@RequestBody QuotaCreateRequest request) {
        return applicationService.createQuota(
                Quota.builder()
                        .productId(request.getProductId())
                        .quantity(request.getQuantity()).status("create").build());
    }

    @Override
    public QuotaItemCreateResponse createQuotaItem(@RequestBody QuotaItemCreateRequest request) {
        QuotaItemCreateCommand command = QuotaItemMapper.MAPPER.toCommand(request);
        return buildResponse(applicationService.createQuotaItem(command));
    }

    private QuotaItemCreateResponse buildResponse(Quota quota) {
        return QuotaItemCreateResponse.builder()
                .quotaId(quota.getId())
                .quantity(quota.getQuantity())
                .quotaItemDtoList(QuotaItemMapper.MAPPER.toResponse(quota.getItems()))
                .build();
    }
}

ApplicationService:
@Service
public class QuotaApplicationService {
    private final QuotaRepository repository;

    public QuotaApplicationService(QuotaRepository repository) {
        this.repository = repository;
    }

    public QuotaCreateResponse createQuota(Quota quota) {
        Quota result = repository.createQuota(quota);

        return QuotaMapper.MAPPER.toResponse(result);
    }

    public Quota createQuotaItem(QuotaItemCreateCommand command) {
        Optional<Quota> optionalQuota = repository.searchQuota(command.getQuotaId());
        if (!optionalQuota.isPresent()) {
            throw new RuntimeException("quota does not exist, id is: "+ command.getQuotaId());
        }

        Quota quota = optionalQuota.get();
        quota.update(command.getQuantity(), command.getQuotaItemDtoList());

        repository.updateQuota(quota);

        return quota;
    }
}

重构之后,ApplicationService中createQuotaItem的实现清晰地分为三部分:
1.查询存在的quota
2.领域模型quota自己完成update操作
3.持久化

领域模型不再是贫血模型,完成它本该完成的更新模型数据的职能。

很多开发人员认为DDD很难,编码时候更是无从下手。思路是DDD的思路,一上手还是三层架构风格的代码。其实很多情况下是领域模型的抽象不够准确,领域模型只是对数据库表结构的翻译。再加上代码职责不够单一,controller和application service的职责混乱,这样的代码自然很难说是DDD的。

其实,DDD很简单,首先定义体现业务本来面貌的领域模型,然后domain, controller,application service, domain service以及repository,每个组件完成自己该干的事情。

云在青天水在瓶,该怎么样就是怎么样。

其实这么说也不太负责,因为抽象一个"体现业务本来面貌的领域模型"本来就是DDD中最为核心且最有难度的事情。

经常思考互联网发展到现在出现的一些社交产品对社交关系的建模。

最早出现的同学录,大家要交流只能留言。沟通实时性太差。社交关系显然不是这样的。
QQ群出现后,同学录就逐渐消亡了。在QQ群里,每个人说的话别人都能第一时间看到。
但是,QQ群无法充分体现个体的差异。
所以,后来出现了QQ空间,微博。个体的表达需求满足了,但是沟通的实时性还是差了一些。
再后来,大家都知道,出现了微信。每个人都能在自己的一亩三分地朋友圈展示自己,好友可以第一时间点赞和评论。如果有群体沟通的需求可以建立各种各样的群。看谁不爽还可以屏蔽,拉黑。现实社会本不就是这样的么?
到目前为止,微信是对社交关系建模最为准确的产品。

本文代码可以从这里获取:
https://github.com/worldlxy/refactor-to-ddd

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容