领域驱动设计,让程序员心中有码(四)

#领域驱动设计,让程序员心中有码(四)

----------------------追忆有关分层的古老往事

         我一直认为,程序员也是艺术家,他们撰写的每一行代码,是献给这大好世界的优美诗篇。不同的人,写的代码也许风格迥异。有的,如春风化雨润物无声,有的,如高山流水,曲高和寡,还有的如旱日春雷,先声夺人。而如果说,代码是诗篇,那么代码的分层艺术绝对是最先映入读者眼帘的序幕了。

         分层,一直以来是一个非常经典的软件工程学问题,提到分层,无论是资深或者新入门的开发者,或多或少都有自己的理解。    

         在8年前,我刚参加工作时,有幸参与了比较多项目的研发和维护过程,这些项目的开发者,大多是比我年长几岁的软件开发者。在他们的开发习惯中,往往会倾向于直接在代码中写入用户界面、数据库访问等支持代码,甚至有相当一部分情况下,会把代码写入到用户界面中,这意味着在用户界面层,往往会写入大量的代码,说不定会超过上万行代码。我觉得可以将这种设计,理解为“单层架构“。

        再后来的项目开始有了一点点改变,这个时候的架构,被成为“三层架构“。在.NET开发史上,三层架构成为一种历史悠久的框架,从十年前开始,一直到今天,依然是.NET开发者最为熟练的技术框架。如图所示,三层架构在客户端和数据库之间增加了一个中间层,将有效的业务规则、数据访问等放在业务层中进行处理。界面层主要使用对数据的绑定渲染,再通过数据层实现数据的提交处理。有的开发者说,三层架构通吃一切项目,似乎所有的项目都可以用三层架构来套用。然而在实际工作中,往往会忽略了业务逻辑层,业务逻辑往往会放在界面层或数据层中,最终导致界面层和数据层充斥着臃肿的代码,令人难以维护。

640?wx_fmt=png

 

    

       MVC架构也是一种非常主流的技术架构,这种架构是一种非常优秀的架构,他将用户界面层通过模型层最早来源于上世纪七十年代,为Smalltalk语言所设计的一种模式,许多基于用户界面的架构也受到了他的启发。MVC架构,通过将逻辑、数据、界面显示分离的形式,实现代码的组织,广泛适用于桌面端或网页程序的开发过程。这种架构的特点是耦合性低、重用度高、生产周期短、部署快。但是同样由于缺乏明确的定义,以及代码过于集中,导致代码腐化问题非常严重。

640?wx_fmt=png

 

    

      MVP架构是从MVC架构中演进出来的典型代表,这种架构通过Presenter来处理逻辑,Model提供数据,View负责显示。当然,他和MVC的区别在于,数据的显示往往是通过Presenter来进行的,所有的界面操作也都是在Presenter中实现,这意味着View层,只是单薄的一层。

640?wx_fmt=png

 

         

    与此同时,开发者们也善于使用存储过程或视图等不同的方式写入复杂的业务逻辑,期待的往往是如果是代码发生变更时,只需对数据库进行修改即可,而无需去修改已经部署在用户端的大量的代码。  使用存储过程、触发器或视图等方式,时至今日,依然是备受关注的一种实现技术,尤其是面向传统产业开发的项目,例如面向制造业的工业控制系统,要进行程序的部署仍然需要非常繁琐的流程,这意味着用户层的部署相对来说不方便,而数据库的变动反而更方便。          

  不同类型的架构或设计的理念迄今仍深刻的影响着开发者的思维模式和工作方法,并最终或多或少的影响着软件工程的实现。          

  从高内聚,低耦合的软件工程学基本原则来说,模块与模块之间的关系,称为耦合,而模块内部的关系称为内聚。而分层的目的,正是为了在满足这个基本原则的前提下,提高代码的可用性和稳定性。过少的分层,意味着代码可读性必然很糟糕,甚至不利于软件的编译分发,而过多的分层,则同样意味着模块的维护将对整体过程代码不便。          

      领域驱动设计认为,如果代码中,与领域相关的内容分散在不同的部分,那么要进行业务和代码的分析将变得难以下手。对用户界面的简单调整都可能意味着代码逻辑的变化,而想要调整业务规则,可能需要开发者们去排查与之相关的所有部分,例如界面层,数据层,甚至是存储过程和视图,这种过程实际上已经没办法建立可用的业务模型,甚至没办法建立测试用例和单元测试的开展,只能靠开发者凭经验去处理不同的情况。为了保证代码的可用性,应当尽可能的分析理清所使用的技术和手段,程序层次设计应当简单明了,易于理解。          

  在复杂系统设计中,分层是一种非常不错的理念,其目的是为了更好的实现关注度分离,使设计的每个部分,既能得到单独的关注,也能维护系统内部复杂的交互关系。领域设计认为,上述提到的几种设计模式,往往是下面四种概念的某种变体。

  •     用户界面层(或者表示层),负责向用户显示信息和解释用户指令。这里的用户,既可以是使用用户界面的人,也可以是另外一个计算机系统。

  •     应用层:定义软件要完成的任务,并且只会表达领域概念的对象来解决问题。这一层实际上负责的是系统与应用层进行交互的必要渠道。

  •     领域层(或模型层)负责表达业务概念、业务状态信息以及业务规则。尽管技术细节由基础设施层实现,但业务情况状态的反映则需要有领域层进行控制。领域层是业务软件的核心。

  •     基础设施层:为上面各层提供通用的技术能力:为应用层传递消息,为领域层提供持久化机制,为用户界面层绘制屏幕组件等等。基础设施层还能够通过架构框架来支持4个层次减的交互模式。

        在传统项目中,往往不见得会如此鲜明的划分层次,但是在进行模型驱动设计的关键,正是需要将领域层从复杂的业务代码中抽离出来。在设计过程中,应当采取以下操作:

    1、给复杂的应用层次划分层次;

    2、对每个层次进行独立的设计,使其具有内聚性并且只依赖于它的下层;

    3、采用标准的架构模式,只与上层进行松散的耦合。

    4、将所有与领域模型有关的代码放在一个层中,并让它与用户界面层、应用层和基础设施层的代码分开。

    5、领域对象应当把重点放在如何表达领域模型上,而不需要考虑自己的显示和存储问题,也无需管理应用任务等内容。

    6、模型设计的目的是达到含义丰富、结构清晰、业务鲜明。

       适当的分层,往往会通过架构框架的形式予以体现。在项目开发过程中,我们通常会使用一些框架,这些框架往往来源于平时项目的积累或者互联网大佬的知识分享。但是这些框架往往不见得完全符合我们的项目需求。领域驱动设计认为,好的架构框架既能解决复杂技术问题,也能让领域开发者集中精力去表达模型,而无需考虑其他问题。然而使用框架很容易为项目制造障碍,最终同样会给人造成错误的理解,即,我们这个项目很简单,并不需要复杂的框架。然而,笔者认为,大部分情形所认为的不需框架的原因,往往是项目确实非常简单,或者开发者低估了项目的复杂度,或者由于公司短视的观点,而忽略了代码的架构设计和分层与建模工作,这最终导致项目容易陷入泥潭。

  当然,也同样要克制欲望,贸然选择难以驾驭的框架,反而让项目开发过程变得不可收拾。因此,要选择性的运用框架来解决问题,明智和审慎的应用框架的价值,让开发者专注于核心业务问题的建模工作,才是提高开发效率和程序质量的手段。

相关文章:

  • 领域驱动设计,让程序员心中有码

  • 领域驱动设计,让程序员心中有码(二)

  • 领域驱动设计,让程序员心中有码(三)

原文地址: https://www.cnblogs.com/xiyuanMore/p/10166790.html

.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com

640?wx_fmt=jpeg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/318296.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

YBTOJ洛谷P4551:最长异或路径(trie树)

洛谷传送门 文章目录题目描述解析代码题目描述 解析 本题关键就在于一点: 若把每个点的深度dep[i]定义为从根到节点边权的异或和 那么i到j的路径异或和可以表示为: dep[i] ^ dep[j] 首先要是i、j在不同子树上显然成立 如果他们在同一子树上,…

P5074-Eat the Trees【插头dp】

正题 题目链接:https://www.luogu.com.cn/problem/P5074 题目大意 给出一个nmn\times mnm的网格,有的必须铺线有的不能,铺成若干条闭合回路,求方案数。 1≤n,m≤121\leq n,m\leq 121≤n,m≤12 解题思路 考虑插头dpdpdp,因为可以…

初赛—算法复杂度

对于 \(T(n) a\times T(\frac{n}{b})c\times n^k\) 这样的递归关系&#xff0c;有这样的结论&#xff1a; \(a>b^k\) 则有 &#xff1a; \(T(n) O(c\times n^{\log_{b} a})\)\(ab^k\) 则有 &#xff1a; \(T(n) O(c\times n^k\times log n)\)\(a<b^k\) 则有 &#xff…

1414. 牛异或

题意&#xff1a; 区间长度为n的数组a&#xff0c;取连续一段子序列&#xff0c;问哪段子序列的异或和最大&#xff0c;如果存在多个这样的序列&#xff0c;那么选择序列末端整数对应的编号更小的那个序列。 如果仍然存在多个可选的序列&#xff0c;那么选择长度最短的那个序列…

基于Asp.Net Core打造轻量级内部服务治理RPC(一)

继之前的《Asp.Net Core Docker 搭建》文章末尾说过的&#xff0c;将陆续编写基于asp.net core 打造一个内部服务治理的rpc框架。不过前端时间较忙&#xff0c;所以搁置了一段时间。闲话不多说&#xff0c;下面就来讲讲为什么需要去做一个该框架&#xff0c;以及想法的来源和设…

P7046-「MCOI-03」诗韵【SAM,倍增,树状数组】

正题 题目链接:https://www.luogu.com.cn/problem/P7046 题目大意 给出一个长度为 nnn 的字符串&#xff0c;然后 mmm 次把它的一个子串加入集合。如果一个字符串在这个集合中作为字符串的后缀出现次数大于 kkk 那么这个字符串就会被计入贡献。 每次求计入贡献的字符串数和最…

YBTOJ:前缀询问(trie树)

文章目录题目描述解析代码题目描述 解析 &#xff08;没有做出来&#xff0c;这个ans的处理方式其实也不难想…qwq&#xff09; 考虑把T都作为模板串加入trie树 加入每个模板串自然就是按照i顺序的 所以我们在插入t的时候沿途标记一下 新出现的未标记的i的间隔就是当前的i与上…

DP专练1( [NOIP 2003]加分二叉树 + 太空梯 )

我们先慢慢来加分二叉树题目题解简单讲解前序//中序//后序遍历代码实现太空梯题目题解代码实现加分二叉树 题目 题解 简单讲解前序//中序//后序遍历 其实说白了&#xff0c;这个*序就是根root的遍历顺序 先序就是root–>left–>right 中序就是left–>root–>rig…

P1896 [SCOI2005]互不侵犯

P1896 [SCOI2005]互不侵犯 题目&#xff1a; 在NN的棋盘里面放K个国王&#xff0c;使他们互不攻击&#xff0c;共有多少种摆放方案。国王能攻击到它上下左右&#xff0c;以及左上左下右上右下八个方向上附近的各一个格子&#xff0c;共8个格子。 题解&#xff1a; 需要知道…

AT2368-[AGC013B]Hamiltonish Path【构造】

正题 题目链接:https://www.luogu.com.cn/problem/AT2368 题目大意 给出 nnn 个点 mmm 条边的一张无向图&#xff0c;然后求一条路径满足 路径长度不小于二。路径无交。对于所有的 xxx 与路径的端点相连&#xff0c;那么 xxx 在路径上。 1≤n,m≤1051\leq n,m\leq 10^51≤n,…

YBTOJ:运动积分(trie树)

文章目录题目描述解析代码题目描述 解析 做了巨长时间… 进行了一次刺激的阅读理解竞赛… 感谢whh dalao&#xff01; 那么让我们分析一下这道题 首先我们考虑单个求x选手的q值 不难发现 i 在第j天和x的大小关系只与a[x]与a[i]二进制下不同的最高位k有关 j的第k位与x相同时&am…

年终总结,我为什么离开舒适区?

当圣诞夜过去&#xff0c;也意味着这一年即将结束&#xff0c;迎来崭新的开始&#xff0c;一年时间既短暂&#xff0c;又漫长&#xff0c;当离开人生的舒适区&#xff0c;将迎来一个又一个的挑战。如果说一个人的优秀&#xff0c;取决于他天赋&#xff0c;以及他为之付出的额外…

NC16886 炮兵阵地

题目&#xff1a; n*m个网格&#xff0c;有平原&#xff0c;有山地&#xff0c;平原可以放部队&#xff0c;部队攻击范围如图&#xff08;不受地形影响&#xff09;&#xff08;H为山地&#xff0c;P为平原&#xff09; 题解&#xff1a; 确定状态&#xff1a; 因为每个炮可…

CF891B-Gluttony【构造】

正题 题目链接:https://www.luogu.com.cn/problem/CF891B 题目大意 给出nnn个数字互不相同的一个序列aaa&#xff0c;求它的一个排列bbb&#xff0c;使得选出任意一个1∼n1\sim n1∼n的下标真子集&#xff0c;都有aaa的对应下标和不等于bbb的对应下标和。 1≤n≤22,0≤ai≤10…

DP专练2 (大理石 + [ZJOI 2010]数字计数)

你肯定以为DP专练会有很多题&#xff0c; 但是请考虑一下本仙女的DP码力&#xff0c;一次性能更几个题。。。 来吧&#xff0c;别害怕呀~~ 文章目录大理石题目题解代码实现数字计数题目题解代码实现大理石 题目 林老师是一位大理石收藏家&#xff0c;他在家里收藏了n块各种…

二阶常系数齐次线性递推数列

定义 若数列 \(\{a\}\) 满足 \(a_nc_1a_{n-1}c_2a_{n-2}\) &#xff0c;\(c_1,c_2\) 为常数&#xff0c;就称这种数列为二阶常系数齐次线性递推数列。 求解 加入能够将递推关系式改写为 \((a_n-ka_{n-1})p(a_{n-1}-ka_{n-1})\) 的形式&#xff0c;就可以求出 \(a_n-ka_{n-1}\) …

YBTOJ:斐波拉契(矩阵快速幂)

文章目录题目描述题目描述代码题目描述 题目描述 关键在于如何转化为本题的题目。。。 设 y(1-根号5&#xff09;/2$$ 再令&#xff1a; A(n)xn yn 通过尝试可以发现&#xff0c;A其实就是一个1,3为前两项的斐波拉契数列 则 xnA(n)-yn A的值可以用矩阵快速幂来求 而y是在(-1,…

ABP 框架 数据库底层迁移 Mysql 集群

技术交流&#xff0c;请加QQ群&#xff1a;538327407我的各种github 开源项目和代码&#xff1a;https://github.com/linbin524背景笔者 目前架构的IOT 项目是使用abp 框架作为后台,虽然abp的框架适用于中小型项目框架&#xff0c;但由于架构优美&#xff0c;笔者认为还是可以经…

CF280D-k-Maximum Subsequence Sum【模拟费用流,线段树】

正题 题目链接:https://www.luogu.com.cn/problem/CF280D 题目大意 一个长度为nnn的序列&#xff0c;mmm次操作 修改一个数询问一个区间中选出kkk段不交子段使得和最大 1≤n≤105,1≤m≤2105,1≤k≤201\leq n\leq 10^5,1\leq m\leq 2\times 10^5,1\leq k\leq 201≤n≤105,1≤…

数论练习1 ( 曹冲养猪 + [POJ 2891]Strange Way to Express Integers + 乘法逆元【带证明】)

虽然作业还没有做完&#xff0c;但是我还是放不下它&#xff0c;对此&#xff0c;我只想说&#xff1a; 今天你对作业爱理不理&#xff0c;明天它就让你补到飞起 DP先放放&#xff0c;我们要雨露均沾练习上手&#xff1a;乘法逆元题目题解代码实现曹冲养猪?&#xff08;互质的…