微服务间的应用服务可直接访问,也可通过API网关。由于跨微服务操作,在进行数据新增和修改操作时,注意保证数据一致性。
领域事件驱动
领域事件驱动包括微服务内和微服务之间的事件。微服务内通过事件总线完成聚合之间的异步处理。微服务之间通过MQ完成。异步化的领域事件驱动机制是一种间接的服务访问方式。
当应用服务业务逻辑处理完成后,如果发生领域事件,可调用事件发布服务,完成事件发布。
当接收到订阅的主题数据时,事件订阅服务会调用事件处理领域服务,完成进一步的业务操作。
- 微服务的服务是从领域层逐级向上封装、组合和暴露
基础层
服务形态主要是仓储服务。仓储服务包括接口和实现:
-
仓储接口服务供应用层或领域层服务调用
-
仓储实现服务,完成领域对象的持久化或数据初始化
领域层
领域层实现核心业务逻辑,负责表达领域模型业务概念、业务状态和业务规则。主要的服务形态有实体方法和领域服务。
实体采用充血模型,在实体类内部实现实体相关的所有业务逻辑,实现的形式是实体类中的方法。
实体是微服务的原子业务逻辑单元。在设计时我们主要考虑实体自身的属性和业务行为,实现领域模型的核心基础能力。不必过多考虑外部操作和业务流程,这样才能保证领域模型的稳定性。
业务规则和逻辑校验在领域层。
DDD提倡富领域模型,尽量将业务逻辑归属到实体对象,实在无法归属的部分则设计成领域服务。领域服务会对多个实体或实体方法进行组装和编排,实现跨多个实体的复杂核心业务逻辑。
对于严格分层架构,如果单个实体的方法需要对应用层暴露,则需要通过领域服务封装后才能暴露给应用服务。
应用层
表述应用和用户行为,负责服务的组合、编排和转发,负责处理业务用例的执行顺序以及结果的拼装,负责不同聚合之间的服务和数据协调,负责微服务之间的事件发布和订阅。
通过应用服务对外暴露微服务的内部功能,这样就可以隐藏领域层核心业务逻辑的复杂性以及内部实现机制。
应用层的主要服务形态有:
-
应用服务
-
事件发布
-
订阅服务
应用服务内用于组合和编排的服务,主要来源于领域服务,也可以是外部微服务的应用服务。
应用服务内还可完成
-
安全认证
-
权限校验
-
初步的数据校验
-
分布式事务控制
为了实现微服务内聚合之间的解耦,聚合之间的服务调用和数据交互应通过应用服务来完成。原则上我们应该禁止聚合之间的领域服务直接调用和聚合之间的数据表关联。
用户接口层
前端应用和微服务之间服务访问和数据交换的桥梁。
处理前端发送的Restful请求和解析用户输入的配置文件等,将数据传递给应用层
或获取应用服务的数据后,进行数据组装,向前端提供数据服务。
主要服务形态是Facade服务。
Facade服务分为接口和实现两个部分。完成服务定向,DO与DTO数据的转换和组装,实现前端与应用层数据的转换和交换。
松散分层架构
- 领域层的实体方法和领域服务可直接暴露给应用层和用户接口层,而无需逐级封装。
缺陷
-
易泄露领域层核心业务逻辑
-
当实体方法或领域服务发生变更,由于服务同时被多层服务调用和组合,难以找出哪些上层服务调用和组合了它,不方便通知到所有的服务调用方
-
该分层架构中,实体A的方法在应用层组合后,暴露给用户接口层aFacade。abDomainService领域服务直接越过应用层,暴露给用户接口层abFacade服务。松散分层架构中任意下层服务都可以暴露给上层服务。
严格分层架构
每层服务只能向直接上层提供服务。虽然实体、实体方法和领域服务都在领域层,但实体和实体方法只能暴露给领域服务,领域服务只能暴露给应用服务。
服务如果需要跨层调用,下层服务需要在上层封装后,才可以提供跨层服务。
比如实体方法需向应用服务提供服务时,需先封装成领域服务。
通过封装:
-
可避免泄露核心业务逻辑的实现
-
避免在应用层沉淀过多本属领域层的核心业务逻辑,防止应用层臃肿
-
当服务发生变更时,由于服务只被紧邻上层的服务调用和组合,只需逐级告知直接上层
-
A实体的方法需封装成领域服务aDomainService才能暴露给应用服务aAppService。abDomainService领域服务组合和封装A和B实体的方法后,暴露给应用服务abAppService
=====================================================================
- 数据持久化对象PO(Persistent Object)
与数据库结构一一映射,是数据持久化过程中的数据载体。
- 领域对象DO(Domain Object)
微服务运行时的实体,是核心业务的载体。
- 数据传输对象DTO(Data Transfer Object)
用于前端与应用层或者微服务之间的数据组装和传输,是应用之间数据传输的载体。
- 视图对象VO(View Object)
用于封装展示层指定页面或组件的数据。
基础层
-
入参是DO,内部将DO转化成PO进行数据库的增删改查
-
执行结果用PO去映射,再转化为DO作为基础层的返回值
先建立DO和PO的映射:
-
当DO数据需持久化时,仓储服务会将DO转换为PO
-
当DO需初始化时,仓储服务从数据库获取数据形成PO,并将PO转换为DO
大多数情况下PO和DO一一对应。但也存在DO和PO多对多,在DO和PO数据转换时,需数据重组。
比如时间范围查询时,会有辅助字段,如:beginTime和endTime,PO这怎么处理?我们的处理方式是增删改用PO,查询时候用QueryPO,QueryPO继承了PO并额外增加用于查询的辅助字段(比如时间、集合、模糊查询等)。
- 有的查询功能,比如按照名称查询,查询条件就是name,DTO、DO和PO是一样的,也需要在每一层都去转化一下么?我们把查询时的对象命名为QueryPO,从用户接口层到基础层的入参都是这一个,这样可以么?
是否要做数据转换?主要是考虑解耦,这样各层不必受其它层的数据限制,它类似齿轮,通过数据转换来做适配。如果从前端到后端数据对象都是一样的,用一个对象其实也是可以的。要结合自己的场景来分析。
一般聚合根用的少,很多情况聚合根要作为对象传到基础层。
领域层
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
中…(img-TTi17ZkK-1714757965391)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!