先说说之前几次DDD项目失败的案例,其实也不能算是失败,只是没有领会DDD的思想。

之前的DDD是建立在数据层之上的,首先是每张数据表对应一个数据实体,每个数据实体由泛型的DAO管理,DAO又被数据上下文继承以实现事务,这就构成了数据层,业务代码是写在DataContext中。

数据层:DataEntity <- DataMapper<DataEntity> <- DataContext

在这样的数据层之上构建了“貌似”DDD的领域层,首先由领域对象继承数据上下文以实现序列化和脏读校验等功能,聚合根又继承领域对象以实现级联更新、延迟加载和管理领域对象之间的关联,最后由实现了LINQ查询的仓储对象管理聚合根,这就是当时的领域层。

领域层:DataContext <- Entity <-RootEntity <- Repository

这套架构陆陆续续用了1年左右,在运用的过程中发现DDD除了增加项目的复杂程度之外,没有带来任何好处,这是为什么呢?

直到前段时间我才发现,构建在数据层之上的领域层完全就是一个错误,领域层应当处于整个项目中的最底层,而且完全不用继承IEntity,IRepository之类的接口,正是这些接口绑架了领域模型。

==================分割线==================

经过重新设计的领域层今天终于完工了,处于最底层的仍旧是数据实体,原来的泛型DAO和DataContext全部都移动到了横切层,以类似于切面的方式向领域层提供服务。

数据实体之上是值对象,值对象是完全按照领域需求构建的。数据实体和值对象之间还有一层薄薄的数据转换服务,用于将面向数据库的数据实体转换为面向领域的值对象。
                    
数据层:DataEntity <-> Data Transformation Services <-> ValueObject

数据层之上是领域层,一个领域对象内持有若干个值对象,领域对象负载维护自生的状态,和领域对象处于同一层次的是领域服务,领域服务更倾向与处理业务逻辑,在两者之上有仓储对象,仅负责查询。

领域层:Entity / Service <- Repository

构建在数据层和领域层之外的是横切层,横切层提供轻量级的数据服务,例如事务管理、对象缓存,单例模式的DataMapper。这些服务都是以using的方式提供,即:只能在小范围的函数块内使用,使用过后自动销毁。

横切层:EventManager / CacheManager / DataManager 等。

==================分割线==================

现在的领域层已经处于整个项目的最底层了,ValueObject完全是按照领域的对象的需求构建的,使用起来得心应手。

今天在一个大项目中实践后发现,完全可以不参考UI设计,不参考原型设计,只按照需求说明书设计领域结构,设计完成之后的领域模型居然很神奇的符合UI和原型的需求。

这样的开发流程似乎也很符合TDD的理念,接口先行、之后是测试、再后来是功能、最后才是UI。

原来的数据层现在分散到了3个部分:
1、删除重量级的ORM功能,例如级联更新,延迟加载、连表查询等
2、基本的单表持久化移动到横切层,注意最下面一条虚线,现在数据持久化是通过数据实体发消息给数据存储实现的。
3、缓存管理、事务管理移动到横切层,现在直接缓存领域对象,而不是缓存数据实体

也就是说,现在已经没有传统意义上的数据层了,数据存储都是通过消息机制将更改DataEntity的操作传递到持久化服务,持久化服务只有非常简单的CUD功能,可以连接到Sql,NoSql甚至XML文件。

这正是这张图想表达的内容,仓储层并不需要和数据库进行交互。

我们习惯性的思维,仓储需要接受一个请求,然后查询数据库获得数据实体,再将数据实体转换为领域对象。人总是有惰性的,有时候为了省事,数据实体就直接参与业务逻辑、甚至是返回给UI了。

在这张图上,领域层、应用层、UI层都是访问不到数据实体的,对领域对象进行的任意操作都会经由数据转换服务(防腐层)转换为若干条DataEntity.Change消息,数据存储(持久化服务)复杂侦听这些消息,并连接数据库。

也就是说领域层只需要修改数据,而不需要关心持久化的问题。

http://www.jdon.com/44900

摘抄来自论坛的一些DDD讨论的更多相关文章

  1. http://www.narkii.com/club/forum-46-1.html 纳金学习论坛,主要是讨论一些unty3D方面的事情,技术栈比较前沿,

    http://www.narkii.com/club/forum-46-1.html  纳金学习论坛,主要是讨论一些unty3D方面的事情,技术栈比较前沿,

  2. 拨乱反正:DDD 回归具体的业务场景,Domain Model 再再重新设计

    首先,把最真挚的情感送与梅西,加油! 写在前面 阅读目录: 重申业务场景 Domain Model 设计 后记 上一篇<设计窘境:来自 Repository 的一丝线索,Domain Model ...

  3. DDD 回归具体的业务场景,Domain Model 再再重新设计

    DDD 回归具体的业务场景,Domain Model 再再重新设计 首先,把最真挚的情感送与梅西,加油! 写在前面 阅读目录: 重申业务场景 Domain Model 设计 后记 上一篇<设计窘 ...

  4. csdn论坛页抓取

    抓取csdn论坛 实现功能 获取论坛分类所有链接,并拼接成推荐精华页的完成的链接 获取推荐精华页的帖子状态,赏分,帖子标题,作者,发布时间,回复量,查看量,最后发表时间 置顶内容不爬取,只打印置顶内容 ...

  5. Repository 仓储,你的归宿究竟在哪?(一)-仓储的概念

    写在前面 写这篇博文的灵感来自<如何开始DDD(完)>,很感谢young.han兄这几天的坚持,陆陆续续写了几篇有关于领域驱动设计的博文,让园中再次刮了一阵"DDD探讨风&quo ...

  6. 超全的web开发工具和资源

    首页 新闻 产品 地图 动态 城市 帮助 论坛 关于 登录 注册 · 不忘初心,继续前进,环境云V2接口正式上线 · 环境云测点地图全新改版 · 祝福各位环境云用户中秋快乐!   平台信息 培训互动 ...

  7. 安全协议系列(五)---- IKE 与 IPSec(中)

    在上一篇中,搭建好了实验环境.完整运行一次 IKE/IPSec 协议,收集相关的输出及抓包,就可以进行协议分析.分析过程中,我们将使用 IKE 进程的屏幕输出和 Wireshark 抓包,结合相关 R ...

  8. wcf和web service的区别

    1.WebService:严格来说是行业标准,不是技术,使用XML扩展标记语言来表示数据(这个是夸语言和平台的关键).微软的Web服务实现称为ASP.NET Web Service.它使用Soap简单 ...

  9. 深入理解javascript:揭秘命名函数表达式

    这是一篇转自汤姆大叔的文章:http://www.cnblogs.com/TomXu/archive/2011/12/15/2288411.html 前言 网上还没用发现有人对命名函数表达式进去重复深 ...

随机推荐

  1. ASP.NET MVC Model元数据

    ASP.NET MVC Model元数据(三) 前言 在上篇中我们大概的讲解了Model元数据的生成过程,并没有对Model元数据本身和详细的生成过程有所描述,本篇将会对详细的生成过程进行讲解,并且会 ...

  2. 收缩SQL Server日志不是那么简单

    收缩SQL Server日志不是那么简单的(翻译)   原文地址:http://rusanu.com/2012/07/27/how-to-shrink-the-sql-server-log/ 说明:本 ...

  3. 在一般处理文件中访问Session需要添加IRequiresSessionState

    在IHttpHandler 使用Session 通常我们经常,通过session判定用户是否登录.还有一些临时的.重要的数据也尝尝存放在Session中. 在页面我们很容易的得到Session的值,但 ...

  4. c# in deep 之使用匿名方法的内联委托操作

    匿名方法允许我们指定一个内联委托的操作,为创建委托实例表达式的一部分.其可以对代码进行极度精简,当然可读性变得很差.下面看一个求平方根的例子. List<int> list = new L ...

  5. UNIX基础知识--<<UNIX 环境编程>>读书笔记

    1 shell程序就是位于应用软件与系统调用之间的程序   每个用户登录系统,系统就会为用户分配shell (用户的登录的口令文件 在  /etc/passwd 2 ls filename  运行原理 ...

  6. Python 用IMAP接收邮件

    一.简介IMAP(Internet Message Access Protocol),这个协议与POP一样,也是从邮件服务器上下载邮件到本机,不过IMAP比POP的功能要更加强大些,IMAP除支持PO ...

  7. vim复制

    关于vim复制剪贴粘贴命令的总结   最近在使用vim,感觉很好很强大,但是在使用复制剪切粘贴命令是,碰到了一些小困惑,网上找了一些资料感觉很不全,讲的也不好,遂自己进行实践并总结了. 首先是剪切(删 ...

  8. easyui 通用的datagrid中如何带有查询条件分页

    html 代码 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www ...

  9. VB的写法,关于版本写入注册表

    VB的写法,关于版本写入注册表   '软件都有一个关于我们的菜单,单单实现这个功能并不难,困难的是能够将这些信息写入注册表中,这样每次只要弹出这样的窗口,就从注册表里去搜索.这样是不是极大的节省了资源 ...

  10. 项目检出JRE问题(Unbound classpath container: 'JRE System Library [JavaSE-1.7]' in project 'idweb')

    分类: java 技术2014-07-31 16:51 9864人阅读 评论(13) 收藏 举报 项目从SVN检出到工作空间后报了很多错误,其中很明显就是一些jar的问题,没有相关的jar或版本问题, ...