阅读目录:
在上一篇《一缕阳光:DDD(领域驱动设计)应对具体业务场景,如何聚焦 Domain Model(领域模型)?》博文中,探讨的是如何聚焦领域模型(抛开一些干扰因素,才能把精力集中在领域模型的设计上)?需要注意的是,上一篇我讲的并不是如何设计领域模型(本篇也是)?而是如何聚焦领域模型,领域模型的设计是个迭代过程,不能一概而论,还在路上。
当有一个简单的领域模型用例,完成一个从上而下过程的时候,就需要对领域模型和数据库进行对象关系映射(ORM),首先,在领域驱动设计中,领域模型是活的(具有自己的行为和状态),而映射到数据库中所谓的表是死的(只是一些字段),如何把活的变成死的?又如何把死的变成活的?更重要的是如何保证在这个“死去活来”的过程中,死的和活的是同一个?
转换过程很简单,使用 ORM(对象关系映射)工具就很方便的完成这个“死去活来”的过程,但是有时候我们在这个转换过程中,可能会失去转换对象的本质,以致活的会变成死的,最后转换过程就只有死的变成死的(反复循环)。
在 MessageManager 项目的上一个版本中,主要存在两个领域模型:Messgae 和 User,他们数据库之间的映射关系是一对多的关系,就是一个用户拥有多个消息,但是一个消息只能对应一个用户(发件人或收件人),我们看下领域模型的设计(暂不包含业务逻辑)。
Domain Model-Message:
1 namespace MessageManager.Domain.DomainModel 2 { 3 public class Message : IAggregateRoot 4 { 5 #region 构造方法 6 public Message() 7 { 8 this.ID = Guid.NewGuid().ToString(); 9 } 10 #endregion 11 12 #region 实体成员 13 public string FromUserID { get; set; } 14 public string FromUserName { get; set; } 15 public string ToUserID { get; set; } 16 public string ToUserName { get; set; } 17 public string Title { get; set; } 18 public string Content { get; set; } 19 public DateTime SendTime { get; set; } 20 public bool IsRead { get; set; } 21 public virtual User FromUser { get; set; } 22 public virtual User ToUser { get; set; } 23 #endregion 24 25 #region IEntity成员 26 /// <summary> 27 /// 获取或设置当前实体对象的全局唯一标识。 28 /// </summary> 29 public string ID { get; set; } 30 #endregion 31 } 32 }
Domain Model-User:
1 namespace MessageManager.Domain.DomainModel 2 { 3 public class User : IAggregateRoot 4 { 5 #region 构造方法 6 public User() 7 { 8 this.ID = Guid.NewGuid().ToString(); 9 } 10 #endregion 11 12 #region 实体成员 13 public string Name { get; set; } 14 public virtual ICollection<Message> SendMessages { get; set; } 15 public virtual ICollection<Message> ReceiveMessages { get; set; } 16 #endregion 17 18 #region IEntity成员 19 /// <summary> 20 /// 获取或设置当前实体对象的全局唯一标识。 21 /// </summary> 22 public string ID { get; set; } 23 #endregion 24 } 25 }
乍一看,Message 和 User 领域模型并没有什么问题,只是设计的太贫血(只是包含一些属性字段),抛开业务逻辑,我们看下 Message 和 User 之间的关联,Message 模型中拥有 FromUserID,FromUserName,ToUserID,ToUserName 字段,用来表示和 User 模型的关联,Navigation Properties(导航属性)为:FromUser 和 ToUser,类型为 User,再看一下 User 模型的导航属性:SendMessages 和 ReceiveMessages,类型为 ICollection<Message>,我们如果按照平常的开发模式(脚本驱动模式),这样设计没有一点问题,很方便对 ORM 进行配置:
1 /// <summary> 2 /// Initializes a new instance of <c>MessageConfiguration</c> class. 3 /// </summary> 4 public MessageConfiguration() 5 { 6 HasKey(c => c.ID); 7 Property(c => c.ID) 8 .IsRequired() 9 .HasMaxLength(36) 10 .HasDatabaseGeneratedOption(DatabaseGeneratedOption.None); 11 Property(c => c.FromUserID) 12 .IsRequired() 13 .HasMaxLength(36); 14 Property(c => c.ToUserID) 15 .IsRequired() 16 .HasMaxLength(36); 17 Property(c => c.Title) 18 .IsRequired() 19 .HasMaxLength(50); 20 Property(c => c.Content) 21 .IsRequired() 22 .HasMaxLength(2000); 23 Property(c => c.SendTime) 24 .IsRequired(); 25 Property(c => c.IsRead) 26 .IsRequired(); 27 ToTable("Messages"); 28 29 // Relationships 30 this.HasRequired(t => t.FromUser) 31 .WithMany(t => t.SendMessages) 32 .HasForeignKey(t => t.FromUserID) 33 .WillCascadeOnDelete(false); 34 this.HasRequired(t => t.ToUser) 35 .WithMany(t => t.ReceiveMessages) 36 .HasForeignKey(t => t.ToUserID) 37 .WillCascadeOnDelete(false); 38 }
上面代码表示 Message 的映射配置,如果外键可以为 NULL,则使用 HasOptional 方法,多对对则使用 HasMany 和 WithMany,WillCascadeOnDelete 用来级联删除配置,EntityTypeConfiguration 的具体详细配置,请参照:http://msdn.microsoft.com/zh-cn/data/jj591620.aspx。
上面的设计到底有没有问题?我们来分析一下,首先 User 领域模型中的 SendMessages 和 ReceiveMessages 属性,如果单独作为导航属性,这是没有什么问题的,因为我们可以使用导航属性很方便的进行映射配置(比如上面代码),但是放在领域模型中就有点不伦不类了,User 是一个用户对象,我们不能在它的身上来挂一些属于它的东西,因为这些并不是用户本身所具有的,这就好像我设计一个用户模型,它拥有手机,电脑,背包,房子,车子等等,然后就必须在这个用户模型中添加这个属性,这样设计就会很不合理,这个应该设计在它所拥有的物品上,因为只有这些物品拥有用户,这些物品相对于用户来说,才有真正的存在意义。
再来看 Message 领域模型,首先 FromUserID,FromUserName,ToUserID,ToUserName 这四个字段就让我们看得很不顺眼,因为这些都是已死的字段,Message 应该关联的是活的 User,而并不是在它身上打上几个 User 的标签,这个表现应该在数据库中(因为数据库中就是存的这些已死的字段),而并不是在活的 Message 领域模型中,FromUser 和 ToUser 的设计是没有问题的,因为关联的就是活的 User 对象。
为什么有了 FromUser 和 ToUser 对象,Message 领域模型中还要添加上面那四个字段呢?主要原因还是受思维模式的影响(脚本驱动模式),虽然是基于领域模型设计,但是在设计过程中就会不自觉的往脚本驱动模式上套,为什么?因为我们要使用数据库,不管怎么设计,这些对象都是要存在数据库中的,而数据库存的都是一些已死的对象(只是包含字段),对象死了,那怎么来表示 Message 和 User 对象之间的关联呢?答案就是 FromUserID 和 ToUserID,因为只有通过这两个字段,才能在数据库中体现 Message 和 User 对象之间的关联,数据库存储中确实是这么做的,但是我们把数据库中的关联表现在领域模型中就很不合适了,最后的结果就是 FromUser 和 ToUser 对象的作用只是用来映射配置,Message 领域模型变成和数据库中的 Message 表一样,状态都是已死,转换也就是死的对象转换为死的对象。
那到底怎么设计?答案就是把 Message 领域模型中的 FromUserID,FromUserName,ToUserID,ToUserName 四个属性去掉,User 领域模型中的 SendMessages 和 ReceiveMessages 属性也去掉,让领域模型变得干净。那有人会问了,你把这些关联字段去掉了,怎么去映射数据库呢?天无绝人之路,使用 EntityFramework(ORM 工具之一)就很方便的进行映射配置,具体配置,可以看下枚举映射和关联映射两个节点。
本节点纯属扯淡,兄台们不感兴趣的话,可以直接略过。
“数据库已死”的这个概念,并不是本人提出的,早在六年前在解道中就有人提出,具体可以参考:
首先,强调一点,数据库已死的概念,并不是说我们项目中不使用数据库(想想应用程序不使用数据库也不可能),只是说应用程序设计的核心不再是基于数据库设计的,而应该是基于面向对象设计,数据库只是存储数据的一种方式,当然也可以配置文件存储或者内存存储。以往我们进行应用程序设计的时候,都是先根据业务需求定义表结构,然后根据表结构用“面向对象”的语言去传递 SQL 放到数据库中执行,这样面向对象语言就成了所谓的 SQL 搬运工,这样造成的问题就是非常难维护,牵一发而动全身,而且性能瓶颈也主要体现在数据库方面,想想应用程序的性能问题(排除代码问题),我们可以使用负载均衡增加服务器,来分担所带来的压力,而应对数据库性能问题呢?从“MySpace”的经历上就可以看出,那是相当的难处理,而且性能问题主要集中在数据库方面,也是设计的不合理所造成的。
我们来看一下 MySqace 的信息系统发展历程:
总结:从 MySpace 看更加验证,数据库是软件系统的瓶颈,而且最不可伸缩,一旦数据库成为系统瓶颈,就得动大手术,实现架构上的变迁,这是伤筋动骨,变迁人员压力巨大的。另外由于是社区,就是变迁数据丢失也没什么大不了,如果是企业那就......
如果我们从软件系统开始之初,就使用对象分析设计,不与数据库沾边,整个流程就完全 OO,分析设计直至代码都摆脱了数据库影响,这个流程如下:
那么数据库在什么时候建立呢?数据库表结构的创建可以延缓到部署运行时,这样,整个上游环节就不涉及数据库技术,而是使用更符合自然的表达 OO 方式,软件质量就更高了。现在,很多人已经理解,分析设计要用 OO,但是数据库是运行阶段缺少不了的,确实,这是正确观点,我们夺取数据库的王位,不是将它打倒,只是理性和平移交权力重心而已,数据库退出主角地位,让位于中间件,也预示着过去数据库为王的时代的结束, 但是数据库会和操作系统一样,成为我们现代软件系统一个不可缺少重要的基础环节。
了解了这么多,回到”设计误区“这一节点,你会发现,造成设计误区的主要原因还是,在设计的时候不自觉以数据库为中心了,而并非领域模型。
在 Message 领域模型中,有个 MessageState 枚举类型,用来表示消息的状态,当然我们也可以使用 Bool 类型的字段来表示,但是消息状态是消息本身的一种状态,用对象来表示更为合适,MessageState 枚举定义如下:
1 namespace MessageManager.Domain.DomainModel 2 { 3 public enum MessageState 4 { 5 Read, 6 NoRead 7 } 8 }
我们使用单元测试对映射转换进行测试,也就是 Code First 模式,测试代码:
1 namespace MessageManager.Repositories.Tests 2 { 3 public class UserRepositoryTest 4 { 5 [Fact] 6 public void AddUserRepository() 7 { 8 IUserRepository userRepository = new UserRepository(new EntityFrameworkRepositoryContext()); 9 User user1 = new User("小菜"); 10 User user2 = new User("大神"); 11 userRepository.Add(user1); 12 userRepository.Add(user2); 13 userRepository.Context.Commit(); 14 } 15 } 16 }
生成数据库发生异常:
这个主要原因是当前 EntityFramework 版本不支持枚举类型映射,当前使用的 EntityFramework 版本为 4.3.1:
1 <?xml version="1.0" encoding="utf-8"?> 2 <packages> 3 <package id="EntityFramework" version="4.3.1" targetFramework="net40" /> 4 </packages>
EntityFramework 的版本太老了,更新版本为 6.1.1,NuGet 更新命令:update-package EntityFramework
EntityFramework 从 4.3.1 升级到 6.1.1 更改的地方(http://msdn.microsoft.com/en-us/data/upgradeef6.aspx):
在升级完 EntityFramework 版本后,重新运行单元测试,但是发现又报如下错误:
解决方案:
在 MessageManagerDbContext 构造函数中添加如下代码:
1 public MessageManagerDbContext() 2 : base("MessageManagerDB") 3 { 4 var ensureDLLIsCopied = System.Data.Entity.SqlServer.SqlProviderServices.Instance; 5 this.Configuration.LazyLoadingEnabled = true; 6 }
重新运行单元测试,测试成功,就会发现在 MessageManagerDB 数据库的 Messages 表中已生成 State 字段,类型为 Int,当然也可以通过 EntityTypeConfiguration 中的 HasColumnType 进行自定义字段类型。
先看一下,如果我们没有进行任何的 EntityTypeConfiguration 关联设置,生成数据库会是怎样?MessageConfiguration 和 UserConfiguration 配置如下:
1 public class MessageConfiguration : EntityTypeConfiguration<Message> 2 { 3 /// <summary> 4 /// Initializes a new instance of <c>MessageConfiguration</c> class. 5 /// </summary> 6 public MessageConfiguration() 7 { 8 HasKey(c => c.ID); 9 Property(c => c.ID) 10 .IsRequired() 11 .HasMaxLength(36) 12 .HasDatabaseGeneratedOption(DatabaseGeneratedOption.None); 13 Property(c => c.Title) 14 .IsRequired() 15 .HasMaxLength(50); 16 Property(c => c.Content) 17 .IsRequired() 18 .HasMaxLength(2000); 19 Property(c => c.SendTime) 20 .IsRequired(); 21 } 22 }
1 /// <summary> 2 /// Represents the entity type configuration for the <see cref="Customer"/> entity. 3 /// </summary> 4 public class UserConfiguration : EntityTypeConfiguration<User> 5 { 6 #region Ctor 7 /// <summary> 8 /// Initializes a new instance of <c>UserConfiguration</c> class. 9 /// </summary> 10 public UserConfiguration() 11 { 12 HasKey(c => c.ID); 13 Property(c => c.ID) 14 .IsRequired() 15 .HasMaxLength(36) 16 .HasDatabaseGeneratedOption(DatabaseGeneratedOption.None); 17 Property(c => c.Name) 18 .IsRequired() 19 .HasMaxLength(20); 20 } 21 #endregion 22 }
上面代码中我们并没有进行关联配置,生成 MessageManagerDB 数据库中 Messages 表结构:
可以看到我们虽然没有进行任何的关联设置,Code First 会自动为我们创建外键关联,仅仅是在 Message 领域模型中添加:
1 public virtual User SendUser { get; set; } 2 public virtual User ReceiveUser { get; set; }
以上效果是我们想要的,这也是 EntityFramework 的进步之处,符合领域驱动设计的思想,领域模型中没有数据库中所谓的主外键关联,有的只是对象之间的关联,而数据库只是存储数据的一种表现,这样数据库设计的概念就不存在了,也让我们忘了数据库的存在,而把更多的精力放在领域模型的设计上,这就是领域驱动设计关键所在。
除了 EntityFramework 默认生成关联配置,我们也可以进行自定义配置,比如,上面生成外键字段为:SendUser_ID 和 ReceiveUser_ID,也可以自定义字段名称:
1 HasRequired(x => x.SendUser) 2 .WithMany() 3 .Map(x => x.MapKey("SendUserID")) 4 .WillCascadeOnDelete(false); 5 HasRequired(x => x.ReceiveUser) 6 .WithMany() 7 .Map(x => x.MapKey("ReceiveUserID")) 8 .WillCascadeOnDelete(false);
上面就是自定义外键字段为:SendUserID 和 ReceiveUserID,关于 EntityTypeConfiguration 的配置,比如一对一,一对多,多对多,联合主外键等等,
文章浏览阅读1k次。通过使用ajax方法跨域请求是浏览器所不允许的,浏览器出于安全考虑是禁止的。警告信息如下:不过jQuery对跨域问题也有解决方案,使用jsonp的方式解决,方法如下:$.ajax({ async:false, url: 'http://www.mysite.com/demo.do', // 跨域URL ty..._nginx不停的xhr
文章浏览阅读2k次。关于在 Oracle 中配置 extproc 以访问 ST_Geometry,也就是我们所说的 使用空间SQL 的方法,官方文档链接如下。http://desktop.arcgis.com/zh-cn/arcmap/latest/manage-data/gdbs-in-oracle/configure-oracle-extproc.htm其实简单总结一下,主要就分为以下几个步骤。..._extproc
文章浏览阅读1.5w次。linux下没有上面的两个函数,需要使用函数 mbstowcs和wcstombsmbstowcs将多字节编码转换为宽字节编码wcstombs将宽字节编码转换为多字节编码这两个函数,转换过程中受到系统编码类型的影响,需要通过设置来设定转换前和转换后的编码类型。通过函数setlocale进行系统编码的设置。linux下输入命名locale -a查看系统支持的编码_linux c++ gbk->utf8
文章浏览阅读750次。今天准备从生产库向测试库进行数据导入,结果在imp导入的时候遇到“ IMP-00009:导出文件异常结束” 错误,google一下,发现可能有如下原因导致imp的数据太大,没有写buffer和commit两个数据库字符集不同从低版本exp的dmp文件,向高版本imp导出的dmp文件出错传输dmp文件时,文件损坏解决办法:imp时指定..._imp-00009导出文件异常结束
文章浏览阅读143次。当下是一个大数据的时代,各个行业都离不开数据的支持。因此,网络爬虫就应运而生。网络爬虫当下最为火热的是Python,Python开发爬虫相对简单,而且功能库相当完善,力压众多开发语言。本次教程我们爬取前程无忧的招聘信息来分析Python程序员需要掌握那些编程技术。首先在谷歌浏览器打开前程无忧的首页,按F12打开浏览器的开发者工具。浏览器开发者工具是用于捕捉网站的请求信息,通过分析请求信息可以了解请..._初级python程序员能力要求
文章浏览阅读7.6k次,点赞2次,收藏6次。@Service标注的bean,类名:ABDemoService查看源码后发现,原来是经过一个特殊处理:当类的名字是以两个或以上的大写字母开头的话,bean的名字会与类名保持一致public class AnnotationBeanNameGenerator implements BeanNameGenerator { private static final String C..._@service beanname
文章浏览阅读6.9w次,点赞73次,收藏463次。1.前序创建#include<stdio.h>#include<string.h>#include<stdlib.h>#include<malloc.h>#include<iostream>#include<stack>#include<queue>using namespace std;typed_二叉树的建立
文章浏览阅读7.1k次。在Asp.net上使用Excel导出功能,如果文件名出现中文,便会以乱码视之。 解决方法: fileName = HttpUtility.UrlEncode(fileName, System.Text.Encoding.UTF8);_asp.net utf8 导出中文字符乱码
文章浏览阅读2.1k次,点赞4次,收藏23次。第一次实验 词法分析实验报告设计思想词法分析的主要任务是根据文法的词汇表以及对应约定的编码进行一定的识别,找出文件中所有的合法的单词,并给出一定的信息作为最后的结果,用于后续语法分析程序的使用;本实验针对 PL/0 语言 的文法、词汇表编写一个词法分析程序,对于每个单词根据词汇表输出: (单词种类, 单词的值) 二元对。词汇表:种别编码单词符号助记符0beginb..._对pl/0作以下修改扩充。增加单词
文章浏览阅读773次。我在使用adb.exe时遇到了麻烦.我想使用与bash相同的adb.exe shell提示符,所以我决定更改默认的bash二进制文件(当然二进制文件是交叉编译的,一切都很完美)更改bash二进制文件遵循以下顺序> adb remount> adb push bash / system / bin /> adb shell> cd / system / bin> chm..._adb shell mv 权限
文章浏览阅读6.8k次,点赞12次,收藏125次。1. 单目相机标定引言相机标定已经研究多年,标定的算法可以分为基于摄影测量的标定和自标定。其中,应用最为广泛的还是张正友标定法。这是一种简单灵活、高鲁棒性、低成本的相机标定算法。仅需要一台相机和一块平面标定板构建相机标定系统,在标定过程中,相机拍摄多个角度下(至少两个角度,推荐10~20个角度)的标定板图像(相机和标定板都可以移动),即可对相机的内外参数进行标定。下面介绍张氏标定法(以下也这么称呼)的原理。原理相机模型和单应矩阵相机标定,就是对相机的内外参数进行计算的过程,从而得到物体到图像的投影_相机-投影仪标定
文章浏览阅读2.2k次。文章目录Wayland 架构Wayland 渲染Wayland的 硬件支持简 述: 翻译一篇关于和 wayland 有关的技术文章, 其英文标题为Wayland Architecture .Wayland 架构若是想要更好的理解 Wayland 架构及其与 X (X11 or X Window System) 结构;一种很好的方法是将事件从输入设备就开始跟踪, 查看期间所有的屏幕上出现的变化。这就是我们现在对 X 的理解。 内核是从一个输入设备中获取一个事件,并通过 evdev 输入_wayland