【架构设计 领域驱动开发 一】三层VSDDD

简介: 【架构设计 领域驱动开发 一】三层VSDDD

本篇为架构设计系列的第一篇,希望通过本系列转变自己的软件设计思想。不再只是单纯代码的搬运工,而要有自己的设计想法。本篇也是领域驱动开发的第一篇。

之前第一次接触java的web编程开始,就被书里提前灌输了如下思想:jsp - action - service - serviceimpl - dao - daoimpl - 数据库。好像这就是一套标准一样,后来在教一个学弟写后端的时候,他问你这service也没干嘛啊,你这serviceimpl好像也没干嘛啊,对了,你这hibernate也没啥作用嘛,我直接用jdbc不就行了,然后被我一顿无情鄙视:小伙子,以后你接触大项目就知道了,要按部就班,这都是前人经验,遵守就行了。其实自己也觉得哪里不对劲,这就是所谓的知其然不知其所以然。对于层层调用早已习惯,因为逻辑简单,按部就班,总不会出错嘛,数据被代码一层层送到前端展示,再一层层被反馈给数据库。各层隔离,多好。其实往往是这样,你越不愿意提前动脑子,你以后费的精力就越多。

三层架构之殇

其实对于小型项目,或者简单逻辑,三层往往就够了,比如我之前做的一个参加比赛的作品,只是几句简单sql然后调用数据层层转到前端用echarts显示就非常的简单,对于大型项目来说,就比较麻烦了,如果你的数据表上了一定规模,那么任何一个数据库内容修改动作都会牵一发而动全身,你得修改对应实体,甚至业务逻辑,而如果一个项目经过几波接手流转,更加难以寻踪。以下部分内容提炼自该篇,如有需求,可以访问。

从三层架构迈向领域驱动设计 http://www.cnblogs.com/betterman/p/4408630.html

三层架构是什么

三层架构的结构

大家眼里通用的三层架构应该是这样的:

严格分层架构模式的特点是上层只能访问相邻的下层,其他层次间的调用都不允许。三层架构就是一种严格分层模式,它把职责划分为界面展示、业务逻辑、数据访问三层,还有一个业务实体,前面三层都要依赖它,所以它并不构成一个层。这里有个问题,严格分层架构只能上层访问相邻的下层所以我们时常会遇到这个问题,action想直接从dao拿数据,但非要经过service,或者现在做的.net开发,我必须通过provider来获取,虽然可以提升安全性和使组织架构更严密,但却写了很多重复的代码(空壳调用),这个得依据具体实现场景了。

三层架构的编程思想

三层架构的特点是一种面向过程的编程思想,特点如下:

  1. 业务实体类中基本上只有属性没有方法。 (深表认同,感觉没什么意义,单纯为满足orm)
  2. 业务逻辑层的类基本上只有方法没有属性。 (一系列接口的堆叠)
  3. 将数据表结构映射为业务实体类是一个惯用做法,以至于有人将其称之为“传统”。这样的好处是只需要理解一套模型,能够通过自动化工具从数据表直接生成业务实体,还能够自然而然的通过自动化机制存储与检索业务实体。但对于复杂点的业务,这样做就是绝大部分问题的根源。
  4. 当业务膨胀起来,需要划分模块的时候,我们有个常用的变形:提取一个服务层出来,用来组合模块间的交互,还为业务逻辑层提供了一个防腐层,可以把记录日志、验证权限、处理异常等职责分配给服务层

典型的结构:jsp — action –service —dao 再加个业务实体

上面介绍的就是我们常说的三层架构,由于采用了严格分层模式,用户界面层是绝对不能跨过业务逻辑层调用数据访问层的,同理服务层也是不能调用数据访问层。其实三层架构还有个更准确的名字—-分层贫血领域模型架构,这样就不管多少层都能概括了,但是这名字没有三层那么朗朗上口。前面名字中的领域模型指的是业务实体,贫血意思业务实体中没有或很少方法。

三层架构缺点

工作简单枯燥

从数据库搬到界面或从界面搬到数据库,长期重复枯燥的工作会让人感觉前途无亮

新增功能,需求变化

1,有些功能,它可能牵涉到了所有的数据表,做这样一个功能,首先要把所有关联到的数据表都找出来,理解清楚表与表之间的关系,比较痛苦的是别人写的代码你基本用不上,都要自己从头搞

2,不知道什么时候需求发生变更。

代码复用率低

不能复用那些久经考验的代码,导致重复代码慢慢积累,系统缺陷也比较多

反思三层架构

三层架构的最大问题在于:实际应用中人们喜欢把内存模型和数据库模型保持一致。三层架构的大部分问题都是从这里衍生出来的。

DDD的优势

三层架构到DDD的转变过程

初步动作

优化层次关系–>重构到面向对象设计–>使用DDD相关模式深入重构

上面讲到,三层是“分层贫血领域模型架构”,那么DDD则可称为“分层充血领域模型架构”。

三层到DDD的过程大体是这样的:

  • 首先推翻严格分层的理念,采用松散分层来重新定义服务层(松散分层的意思是上层可以访问下层,而不只是相邻的下层),把调用数据访问层的职责交给服务层,接着把业务逻辑层移动到与业务实体在一起
  • 再接着融合业务逻辑与业务实体,使之成为面向对象的设计,然后利用DDD的模式进行更深入的重构。在DDD技术掌握的还不是那么扎实的时候,三层技术基本仍然能够继续使用。

我们看到业务逻辑与数据访问层已经没有依赖关系了。

由于目前的服务层职责是非常轻的,甚至有很多空壳的调用,所以平衡一下职责,把调用数据访问层的职责从业务逻辑层提升到服务层,需要的数据通过参数传递给业务逻辑层。这样,对于那些简单到无业务逻辑的CRUD就不需要去访问业务层了,直接调用数据访问层。

注意:有些逻辑无法归类到任何一个业务实体上的,就让它留在原地,成为一个领域服务。有些逻辑连领域服务都无法归类的,就让它留在服务层,在坚持大方向的前提下,细节灵活处理,因为每次重构都能够让你对未来的路看到更清楚,很可能下一次重构后你就会为这些流浪逻辑找到合适的家了。

这就是简单模型:基础设施依赖领域层

深化动作

标准分层结构

正式介绍一下DDD的分层架构,当然DDD的架构并不限于分层,现在还有六边形架构、CQRS、EventSourcing等技术可以选用,但是不要好高骛远,先把分层架构掌握好。我准备从标准的结构入手。

各层职责

用户界面层:

原版—-负责向用户展现信息以及解释用户命令。

补充—- MVC中V和C都属于UI层,V展现信息,C解析用户命令。UI像地图一样把各个控制器关联了起来。

应用层:

原版—-很薄的一层,用来协调应用的活动。它不包含业务逻辑。它不保留业务对象的状态,但它保有应用任务的进度状态。

补充—-协调应用的活动这句话太抽象了,我充实一下它:从数据访问中获取领域对象,调用领域对象的方法完成任务,然后再调用数据访问代码把领域对象的改变持久化。事务、权限检查、记录日志、处理异常的职责也归它管。这点和前面三层的服务层的职责其实是一样的。

领域层:

原版—-本层包含关于领域的信息。这是业务软件的核心所在。在这里保留业务对象的状态,对业务对象和它们状态的持久化被委托给了基础设施层。

补充—-业务对象的持久化工作我们已经提升到应用层了,一般情况下,这层最好不要涉及资源库的调用,但是并不绝对。资源库的抽象要么在领域层中,要么提升到了“应用程序框架”,领域层是不会依赖基础设施的。

基础设施层:

原版—-本层作为其他层的支撑库存在。它提供了层间的通信实现对业务对象的持久化,包含对用户界面层的支撑库等作用。

补充—-这层的职责包含了三层的数据访问,但是并不代表UI层就可以调用数据访问的代码,而且职责范围扩充了,有人可能会把它当作存放公用代码的地方,但是建议这里只存放本项目公用的东西,如果能够跨项目公用的代码应该放在一个叫做“应用程序框架”的项目来完成,每个公司都应该有自己的应用程序框架。

回顾一下三层架构与DDD分层的区别:

a. UI层技术基本一样,一些比较智能的绑定可能无法进行了。

b. 服务层基本一样。

d. 业务实体+业务逻辑 = 领域层

e. 如果三层架构不采用业务实体与数据表一致的做法,这层也是一样。

TML标题分割线=======================

开发过程回顾

典型的三层架构开发过程

三层架构以数据库为核心,首先根据需求建立数据模型,然后根据数据模型建立数据库,再根据数据表结构自动生成业务实体,然后可以同时进行业务逻辑及用户界面的开发。

1,一旦开发中业务有所变动,那么首先要修改数据库,重新生成实体,然后其他开发小朋友才能继续工作,网络连接不好、数据库挂机、数据库管理员未能及时更新数据库等等原因都会导致开发过程被打断,好处是终于可以休息一会,坏处是由于要赶进度晚上可能得加班。

2,经常变更数据库结构总不是一件愉快的事情,为了避免数据库结构变动,要求编码阶段之前尽量要把需求固定下来,中后期越少变动越好。由于业务逻辑与数据库绑定紧密,所以单元测试困难重重。当性能出现问题的时候,到底是代码逻辑没有写好,还是Sql语句没有写好,诊断有点困难。由于类的粒度太大,设计模式罕有用武之地,设计原则跟自己没有关系,久而久之可能会觉得编程思想都是骗人的。

DDD开发过程

DDD以领域模型为核心,首先根据需求建立领域模型,然后根据领域模型建立代码模型,然后就可以同时开发业务逻辑和用户界面了。

1,我们有一个模拟数据库存储接口的内存数据库,因此这时候不需要考虑数据库,如果这时业务发生变化,直接修改对象结构是非常简单的事情。直到开发后期,所有功能都已经开发完成,业务结构也稳定下来了,我们再根据领域模型建立数据库,进行一个简单的与业务逻辑集成就完工了。

2,免除了开发过程中数据库经常变动的烦恼,开发过程不会被打断,随时可以进行单元测试,随意可以与用户界面集成把系统跑起来,开发逻辑的时候不考虑数据库,开发数据库的时候不考虑业务,关注点更加集中了。出现性能问题可以单独测试数据库组件和业务逻辑组件,更容易定位问题。类的粒度很小,各种重用,各种设计思想都纷纷派上用场,感觉终于跟国际接轨了。

初识DDD

经过了以上博文的阅读归纳后,话是有些不知所以,觉得隐约确实好一些,但好处体现的似乎很模糊,初步体验就是三点:

1,开发初期邀请领域专家介入,使用通用语言沟通,更好的明确需求与业务。

2,将对数据表驱动开发的模式思路转换为业务,领域驱动开发的模式思路。具体操作就是,先建立领域模型,而不是数据模型,这样便于后期的修改和扩展。

3,将单纯的实体改为领域对象,使用充血模式开发,而不是贫血模式开发,这样不容易导致健忘症。

之后在看一本vaughn Vernon的《实现领域驱动设计》一书,会逐渐更新这方面的博客,争取在自己有架构能力(虽然很遥远)的时候实现直接使用领域驱动开发模式来架构。

相关文章
|
8月前
|
人工智能 运维 安全
配置驱动的动态 Agent 架构网络:实现高效编排、动态更新与智能治理
本文所阐述的配置驱动智能 Agent 架构,其核心价值在于为 Agent 开发领域提供了一套通用的、可落地的标准化范式。
3482 103
|
8月前
|
SQL 前端开发 关系型数据库
如何开发一套研发项目管理系统?(附架构图+流程图+代码参考)
研发项目管理系统助力企业实现需求、缺陷与变更的全流程管理,支持看板可视化、数据化决策与成本优化。系统以MVP模式快速上线,核心功能包括需求看板、缺陷闭环、自动日报及关键指标分析,助力中小企业提升交付效率与协作质量。
|
8月前
|
NoSQL 数据可视化 安全
如何开发一套车辆管理系统?(附架构图+流程图+代码参考)
本文介绍了如何通过搭建车辆管理系统(VMS)帮助企业摆脱传统管理方式,实现流程化、可视化、合规化和自动化。内容涵盖系统架构、关键功能模块、数据模型、API设计、前后端实现及实施建议,提供可落地的技术方案,助力企业降低隐形成本、提升管理效率与透明度,实现数据驱动决策。
|
8月前
|
JSON 文字识别 BI
如何开发车辆管理系统中的加油管理板块(附架构图+流程图+代码参考)
本文针对中小企业在车辆加油管理中常见的单据混乱、油卡管理困难、对账困难等问题,提出了一套完整的系统化解决方案。内容涵盖车辆管理系统(VMS)的核心功能、加油管理模块的设计要点、数据库模型、系统架构、关键业务流程、API设计与实现示例、前端展示参考(React + Antd)、开发技巧与工程化建议等。通过构建加油管理系统,企业可实现燃油费用的透明化、自动化对账、异常检测与数据分析,从而降低运营成本、提升管理效率。适合希望通过技术手段优化车辆管理的企业技术人员与管理者参考。
|
8月前
|
消息中间件 缓存 JavaScript
如何开发ERP(离散制造-MTO)系统中的生产管理板块(附架构图+流程图+代码参考)
本文详解离散制造MTO模式下的ERP生产管理模块,涵盖核心问题、系统架构、关键流程、开发技巧及数据库设计,助力企业打通计划与执行“最后一公里”,提升交付率、降低库存与浪费。
|
7月前
|
前端开发 JavaScript BI
如何开发车辆管理系统中的车务管理板块(附架构图+流程图+代码参考)
本文介绍了中小企业如何通过车务管理模块提升车辆管理效率。许多企业在管理车辆时仍依赖人工流程,导致违章处理延误、年检过期、维修费用虚高等问题频发。将这些流程数字化,可显著降低合规风险、提升维修追溯性、优化调度与资产利用率。文章详细介绍了车务管理模块的功能清单、数据模型、系统架构、API与前端设计、开发技巧与落地建议,以及实现效果与验收标准。同时提供了数据库建表SQL、后端Node.js/TypeScript代码示例与前端React表单设计参考,帮助企业快速搭建并上线系统,实现合规与成本控制的双重优化。
|
8月前
|
消息中间件 JavaScript 前端开发
如何开发ERP(离散制造-MTO)系统中的技术管理板块(附架构图+流程图+代码参考)
本文详解ERP(离散制造-MTO)系统中的技术管理板块,涵盖产品定义、BOM、工序、工艺文件及变更控制的结构化与系统化管理。内容包括技术管理的核心目标、总体架构、关键组件、业务流程、开发技巧与最佳实践,并提供完整的参考代码,助力企业将技术数据转化为可执行的生产指令,提升制造效率与质量。
|
8月前
|
消息中间件 JavaScript 关系型数据库
如何开发一套ERP(离散制造-MTO)系统(附架构图+流程图+代码参考)
本文介绍了面向离散制造-MTO(按订单生产)模式的ERP系统设计与实现方法。内容涵盖ERP系统定义、总体架构设计、主要功能模块解析、关键业务流程(订单到交付、BOM展开、MRP逻辑、排产等)、开发技巧(DDD、微服务、事件驱动)、参考代码示例、部署上线注意事项及实施效果评估。旨在帮助企业与开发团队构建高效、灵活、可扩展的ERP系统,提升订单交付能力与客户满意度。
|
8月前
|
NoSQL 关系型数据库 BI
如何开发一套固定资产管理系统?(附架构图+流程图+代码参考)
固定资产管理涉及采购、入库、维修、盘点、报废等多个环节,是企业资产保值增值的关键。本文详解固定资产管理系统(FAMS)的核心功能、系统架构、资产全生命周期流程,并提供功能设计、开发实操技巧与关键代码示例,涵盖台账、申购、入库、报修、处置、盘点等重点模块。内容聚焦企业落地实践,帮助提升资产管理效率、降低风险、保障审计合规。
|
8月前
|
人工智能 安全 数据可视化
配置驱动的动态Agent架构网络:实现高效编排、动态更新与智能治理
本文系统性地提出并阐述了一种配置驱动的独立运行时Agent架构,旨在解决当前低代码/平台化Agent方案在企业级落地时面临困难,为Agent开发领域提供了一套通用的、可落地的标准化范式。
598 18
配置驱动的动态Agent架构网络:实现高效编排、动态更新与智能治理