DDD分层架构的应用

开篇

  • 上周有机会去上海参加了2021年的架构师峰会,分享中提到了很多云原生的概念,但是云原生的概念更多适合云厂商而非业务相关的开发。作为业务开发的我更关注的是业务相关的分享,也就是这次分享中的DDD相关的概念。

  • DDD有很多概念,包括通用语言、实体、Repository、分层架构等等,这次主要总结的是分层架构和实体定义。DDD是一个仁者见仁智者见智的概念,很多东西都可以直接往上套。

  • 这次分享给我最大的感触就是没有为了DDD而DDD,而是运用其中提到的分层架构来解决实际遇到的问题,而这种解决方案是比较有参考性和落地意义的。


架构规范

  • 业务开发的整体架构规范如上图所示,整体的访问是从左到右。
  • 从架构维度的分层为Interface层、application层、domain层、infrastructure层。
  • Interface层接收和转化参数访问application层,响应application层返回值进行返回。
  • Application层负责串联业务流程,通过Domain层的Service来完成业务处理。
  • Domain层负责以领域模型的划分处理具体的业务。
  • Infrastructure层作为基础设施层,主要封装第三方业务访问包含MQ数据库RPC等。

对比我们日常spring mvc模式的分层架构,我们能够发现本质上MVC就是在践行DDD的分层架构,但不是每个使用mvc分层架构的实现都有明确的分层边界定义,不是每个使用mvc分层架构的实现都定义了每层使用的领域对象。

所以在原来的基础上定义每层的业务边界和数据边界就是更好的应用分层实现业务扩展


分层架构

服务层定义

  • 服务对外提供的接口包括面向web client的rest接口或者面向rpc client的dubbo接口。我们定义Controller为对应的Api Layer(Interface层),Api层负责接收请求并交由Service层进行处理。
  • Service层包含Application和Domain两个维度的Service,Application Service主要是负责业务流程的编排,Domain Service负责具体业务的处理,但是根据具体业务的复杂性我们可能只有Application Service而没有Domain Service,但是建议包含Domain Service。
  • Persist Layer或者Rpc Layer可以映射为Infrastructure层,主要用来访问DB或者外部的rpc服务,为了屏蔽底层差异新增的屏蔽层。

数据层定义

  • Api层对外定义参数变量Param,对Service层转化为DO/BO对象,负责Param到DO/BO的转换。
  • Service层对Api层定义DO/BO对象,对Persist/Rpc层定义PO/DTO对象,对内负责DO/BO到PO的转换,对外负责DO/BO到VO/DTO的转换。
  • Persist/Rpc层对Service定义PO/DTO对象,对外PO/DTO到DO/BO的转换。
  • 不同层定义了的参数之间相互转换通过MapStruct来实现降低成本。
  • 从分层依赖的角度来看上层依赖于下层,因此数据对象定义应该在下层进行定义;上层传给下层的数据由上层负责转换,下层传给上层的数据由下层负责转换。


Api层异常处理

  • 通过切面来解决Spring Mvc返回结果的异常处理,通过Filter来解决Dubbo返回结果的异常处理。
@PostMapping("checkout")
public Result<OrderDTO> checkout(Long itemId, Integer quantity) {
    try {
        CheckoutCommand cmd = new CheckoutCommand();
        OrderDTO orderDTO = checkoutService.checkout(cmd);    
        return Result.success(orderDTO);
    } catch (ConstraintViolationException cve) {
        // 捕捉一些特殊异常,比如Validation异常
        return Result.fail(cve.getMessage());
    } catch (Exception e) {
        // 兜底异常捕获
        return Result.fail(e.getMessage());
    }
}
  • Controller层所有Api接口都有重复的异常处理逻辑,代码重复率极高。
  • 通过自定义注解和切面来优化代码逻辑,Api层只包含正常的业务处理流程。
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ResultHandler {

}

@Aspect
@Component
public class ResultAspect {
    @Around("@annotation(ResultHandler)")
    public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable {
        Object proceed = null;
        try {
            proceed = joinPoint.proceed();
        } catch (ConstraintViolationException cve) {
            return Result.fail(cve.getMessage());
        } catch (Exception e) {
            return Result.fail(e.getMessage());
        }
        return proceed;
    }
}
  • 切面统一封装异常的处理的流程。


总结

  • 目前我对于DDD的理解更多是在分层的理念上,从外到层依次氛围Api层、Application Service层、Domain Service层、Repository层、Infrastructure层。

  • 相比较传统spring mvc的三层架构,为了分层的更加明确性DDD做了更多的细化。针对Service层拆分为了Application Service层和Domain Service层,Domain Service负责处理特定域的服务,Application Service层负责通过Domain Service进行业务流程编排;针对数据访问层或者外部的Rpc服务调用层,我们增加了Repository来进行依赖隔离。

  • 明确了每层的数据边界和数据转换的方法,各层数据包含Param+DO+VO+PO等。通过MapStruct降低数据转换的成本,也利于提高我们分层定义对象的积极性。


参考

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