一、前言

结合我们本次系列的第一篇博文中提到的上下文映射图(传送门:如何一步一步用DDD设计一个电商网站(一)—— 先理解核心概念),得知我们这个电商网站的核心域就是销售子域。因为电子商务是以信息网络技术为手段,以商品交换为中心的商务活动,一个好的核心域设计可以大大提升企业的竞争力和对市场变化的相应速度。

 那么我们开始设计领域对象。对于设计领域对象的基本概念不了解的可以先阅读我的该系列第二篇文章(传送门:如何一步一步用DDD设计一个电商网站(二)—— 项目架构)。

二、定义几个基类

我相信我们大部分人会以如下的方式去存放我们定义的基类,见图1。

              【图1】

这是一种比较常规的技术分层思维方式产生的结果,在某些项目文件中或多或少有那么几个"Base"、"Core"、"Common"等的文件夹存放着一些通用的类,它们起着对当前项目中类的抽象、实现通用性支撑性功能的作用。然而在DDD中这些都应属于基础设施层的事情,这样能够保证其他层专注于自身的职责,不会把本应内聚的东西泄露到这些类中。如我们当前的领域层就专注于领域建模,里面的概念全部与通用语言相关。说干就干,搬到基础设施层去,再取个能表达出一致概念的名字的模块存放,如图2。

             【图2】

三、核心域(销售子域)中有什么

“销售”用通俗的话讲就是“把商品卖给用户”,这几个字中就已经凸显出几个概念:“商品”,“用户”,“卖”。以下就是”商品“和”用户“的代码实现:

            【图3】

              【图4】

我相信有许多人会以图3和图4的方式定义我们的商品和用户,乍一看的确符合对商品、用户概念的独立的理解。但在DDD中有不同的限界上下文,每个限界上下文专注处理自身的业务,多个限界上下文之间是以协作的方式工作。保证多个限界上下文之间的良好协作关系的方式是提高自治性。提高自治性的方式又有很多,技术方面如领域事件、消息队列、事件源等,这里暂时不展开描述。从代码层面来看,建模的时候只获取对当前上下文业务处理刚刚好大小的数据,也可以提高当前项目的自治性。

根据我们划分的上下文映射图,用户和商品是属于另外的上下文的,那么在这里我们都应该建模为值对象,因为我们是无法直接修改这些对象的内部属性的。另外,在上面2个图中,获取其他上下文中的资源时,我们作为客户方基本上是不会原封不动的消费服务方提供的数据的。比如这里面的Product.PermitNo(批文号),在大部分行业里,它与销售商品没什么关系。再如User.BlockedBalance(冻结余额),在销售的过程中,只需要知道用户有多少可用余额就好了。这个思路总结一下就是,从业务角度我们不求大而全,只求刚好满足当前业务即可,但是当业务发生变化的时候我们的领域模型也要及时反映出调整后的通用语言概念。得到修改后的模型:

                  【图5】

                  【图6】

对了,不管是值对象还是实体和聚合,习惯在构造函数中的做好守卫验证,有利于表达出什么样的领域对象是合法的。

四、更好的存活于分布式背景下

 在某些背景下一个限界上下文是作为独立的服务对外提供API进行访问的,特别在电商行业,分布式系统的构建是个普遍情况,方式也多元化,各种RPC框架、Restful等技术选型,SOA、微服务等实现理念层出不穷。如何最大化的降低技术变更和业务变化导致的上下文划分调整的影响,也是我们要考虑的重要问题。

 对于我们.Net开发人员来说,在分布式场景下用的最多的方式无非是WebAPI和WCF了。这种方式也就是在第一篇文章中所提到的发布语言和开放主机服务,那么对于客户端来说需要做好防腐层(第一篇文章中有提到)的工作,好避免外部上下文的概念侵入到自身的领域概念中。一个普遍的防腐层实现时序图,其中真正负责防腐层工作的是XXXAdapter和XXXTranslator,如下(摘自[Vaughn Vernon]《实现领域驱动设计》):

                        【图7】

我们这里实现的相关类如下图所示定义:

                  【图8】

  其中1存放着访问远程资源的接口定义,2是其实现方式。这样设计的好处是,对于领域层的建模隐藏了数据获取的实现细节。并且当我们实际开发的时候可能由于需要配合服务方还未准备好,但是这丝毫不影响我们的开发工作,我们可以定义个Mock类来实现这里的IRemoteServices中的接口,就可以顺利地进行开发工作。

  其中ProductAdapter、UserAdapter分别负责请求商品上下文和用户上下文并取得原始结果,ProductTranslator、UserTranslator则是通过解析原始结果,转换为我方上下文中需要的领域模型概念。以下则是核心部分的实现:

                  【图9】

                  【图10】

五、结语
 本来打算想把整个“把商品卖给用户”过程讲完,但是发现这样篇幅太长,所以这次就先到这里,先把其中的2个主体整个设计过程给弄明白了,接下去再讲“卖”这个行为该如何设计。谢谢各位看官:)
 本文的源码地址:https://github.com/ZacharyFan/DDDDemo,注意其中的Restful请求地址和Json解析类为假实现,仅为了用于支持本文的表述。
    

作者:Zachary_Fan
出处:http://www.cnblogs.com/Zachary-Fan/p/6036729.html

如何一步一步用DDD设计一个电商网站(三)—— 初涉核心域的更多相关文章

  1. 如何一步一步用DDD设计一个电商网站(九)—— 小心陷入值对象持久化的坑

    阅读目录 前言 场景1的思考 场景2的思考 避坑方式 实践 结语 一.前言 在上一篇中(如何一步一步用DDD设计一个电商网站(八)—— 会员价的集成),有一行注释的代码: public interfa ...

  2. 如何一步一步用DDD设计一个电商网站(八)—— 会员价的集成

    阅读目录 前言 建模 实现 结语 一.前言 前面几篇已经实现了一个基本的购买+售价计算的过程,这次再让售价丰满一些,增加一个会员价的概念.会员价在现在的主流电商中,是一个不大常见的模式,其带来的问题是 ...

  3. 如何一步一步用DDD设计一个电商网站(十)—— 一个完整的购物车

     阅读目录 前言 回顾 梳理 实现 结语 一.前言 之前的文章中已经涉及到了购买商品加入购物车,购物车内购物项的金额计算等功能.本篇准备把剩下的购物车的基本概念一次处理完. 二.回顾 在动手之前我对之 ...

  4. 如何一步一步用DDD设计一个电商网站(七)—— 实现售价上下文

    阅读目录 前言 明确业务细节 建模 实现 结语 一.前言 上一篇我们已经确立的购买上下文和销售上下文的交互方式,传送门在此:http://www.cnblogs.com/Zachary-Fan/p/D ...

  5. 如何一步一步用DDD设计一个电商网站(六)—— 给购物车加点料,集成售价上下文

    阅读目录 前言 如何在一个项目中实现多个上下文的业务 售价上下文与购买上下文的集成 结语 一.前言 前几篇已经实现了一个最简单的购买过程,这次开始往这个过程中增加一些东西.比如促销.会员价等,在我们的 ...

  6. 如何一步一步用DDD设计一个电商网站(五)—— 停下脚步,重新出发

    阅读目录 前言 单元测试 纠正错误,重新出发 结语 一.前言 实际编码已经写了2篇了,在这过程中非常感谢有听到观点不同的声音,借着这个契机,今天这篇就把大家提出的建议一个个的过一遍,重新整理,重新出发 ...

  7. 如何一步一步用DDD设计一个电商网站(四)—— 把商品卖给用户

    阅读目录 前言 怎么卖 领域服务的使用 回到现实 结语 一.前言 上篇中我们讲述了“把商品卖给用户”中的商品和用户的初步设计.现在把剩余的“卖”这个动作给做了.这里提醒一下,正常情况下,我们的每一步业 ...

  8. 如何一步一步用DDD设计一个电商网站(十一)—— 最后的准备

     阅读目录 前言 准备 实现 结语 一.前言 最近实在太忙,上周停更了一周.按流程一步一步走到现在,到达了整个下单流程的最后一公里——结算页的处理.从整个流程来看,这里需要用户填写的信息是最多的,那么 ...

  9. 如何一步一步用DDD设计一个电商网站(十二)—— 提交并生成订单

    阅读目录 前言 解决数据一致性的方案 回到DDD 设计 实现 结语 一.前言 之前的十一篇把用户购买商品并提交订单整个流程上的中间环节都过了一遍.现在来到了这最后一个环节,提交订单.单从业务上看,这个 ...

随机推荐

  1. Fis3前端工程化之项目实战

    Fis3项目 项目目录结构: E:. │ .gitignore │ fis-conf.js │ index.html │ package.json │ README.md │ ├─material │ ...

  2. 结巴分词3--基于汉字成词能力的HMM模型识别未登录词

    作者:zhbzz2007 出处:http://www.cnblogs.com/zhbzz2007 欢迎转载,也请保留这段声明.谢谢! 1 算法简介 在 结巴分词2--基于前缀词典及动态规划实现分词 博 ...

  3. MVC5 网站开发之九 网站设置

    网站配置一般用来保存网站的一些设置,写在配置文件中比写在数据库中要合适一下,因为配置文件本身带有缓存,随网站启动读入缓存中,速度更快,而保存在数据库中要单独为一条记录创建一个表,结构不够清晰,而且读写 ...

  4. Carousel 旋转画廊特效的疑难杂症

    疑难杂症 该画廊特效的特点就是前后元素有层级关系. 我想很多人应该看过或者用过这个插件carousel.js,网上也有相关的教程.不知道这个插件的原型是哪个,有知道的朋友可以告诉我. 该插件相对完美, ...

  5. Hawk 5. 数据库系统

    Hawk在设计之初,就是以弱schema风格定义的.没有严格的列名和列属性.用C#这样的静态强类型语言编写Hawk,其实并不方便.但弱schema让Hawk变得更灵活更强大. 因此,Hawk虽然之前支 ...

  6. WebGIS中等值面展示的相关方案简析

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 等值面是气象.环保等相关项目上常用到的效果展示.在传统的CS项 ...

  7. dedecms 后台栏目添加图片

    前台调用栏目时需要显示图标,整理一下: 第一步:“系统->SQL命令工具” , 插入sql语句 alter table dede_arctype add typeimg varchar() 第二 ...

  8. arcpy+PyQt+py2exe快速开发桌面端ArcGIS应用程序

    前段时间有一个项目,大体是要做一个GIS数据处理工具. 一般的方法是基于ArcObjects来进行开发,因为我对ArcObjects不太熟悉,所以就思考有没有其他简单快速的方法来做. 在查看ArcGI ...

  9. 大数据之Yarn——Capacity调度器概念以及配置

    试想一下,你现在所在的公司有一个hadoop的集群.但是A项目组经常做一些定时的BI报表,B项目组则经常使用一些软件做一些临时需求.那么他们肯定会遇到同时提交任务的场景,这个时候到底如何分配资源满足这 ...

  10. MySQL 数据库双向同步复制

    MySQL 复制问题的最后一篇,关于双向同步复制架构设计的一些设计要点与制约. 问题和制约 数据库的双主双写并双向同步场景,主要考虑数据完整性.一致性和避免冲突.对于同一个库,同一张表,同一个记录中的 ...