关于模型的合法性,Entity.IsValid()合理吗?
关于模型的合法性,Entity.IsValid()合理吗?
背景
见过很多框架(包括我自己的)都会在实体的定义中包含一个IsValid()方法,用来判断实体的合法性,是否应该这样设计呢?本文就这个问题介绍一点想法,希望大家多批评。
实体能否处于“非法”状态?
实体是否应该包含IsValid()方法的深层次问题是:“实体能否处于非法状态?”。我们来定义一些术语,接下来我就引用这些术语:
- A模式:实体允许处于非法状态,但是实体要包含一个IsValid()方法进行校验。
- B模式:实体不允许处于非法状态,业务逻辑必须保证这一点。
关于A模式我不想多说了,A模式本身没有问题的,今天重点说说如何实现B模式。
如何实现B模式?
最好的说明就是写一个例子,下面是我们例子的需求:
- xxx属性不能为空。
- xxx属性必须唯一。
这个例子非常简单,也具有代表性,可以进一步抽象为:
- xxx属性不能为空,聚合自身的验证。
- xxx属性必须唯一,跨聚合验证。
让我们一个一个来。
xxx属性不能为空,聚合自身的验证。
聚合本身应该负责自己状态的完整性,反射可能会绕过这些验证,使用类似AutoMapper的工具需要注意(我已经处理了)。

1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6
7 using Happy.Domain;
8 using Happy.Domain.Tree;
9 using Happy.Example.Events.TestGrids;
10
11 using Happy.Infrastructure;
12
13 namespace Happy.Example.Domain.TestGrids
14 {
15 public partial class TestGrid : AggregateRoot<Guid>
16 {
17 public System.Int64? BigIntField { get; set; }
18 public System.Boolean? BitField { get; set; }
19 public System.DateTime? DateField { get; set; }
20 public System.DateTime? DateTimeField { get; set; }
21 public System.Decimal? DecimalField { get; set; }
22 public System.Double? FloatField { get; set; }
23 public System.Int32? IntField { get; set; }
24 public System.Decimal? MoneyField { get; set; }
25 public System.Decimal? NumericField { get; set; }
26 public System.String NVarcharField { get; private set; }
27 public System.Single? RealField { get; set; }
28 public System.TimeSpan? TimeField { get; set; }
29 public System.Byte[] TimestampField { get; set; }
30
31 public void SetNVarcharField(string value)
32 {
33 value.MustNotNullAndNotWhiteSpace(value);
34
35 this.NVarcharField = value;
36 }
37
38 internal void PublishCreatedEvent()
39 {
40 this.PublishEvent(new TestGridCreatedEvent());
41 }
42 }
43 }

xxx属性必须唯一,跨聚合验证。
仓储负责判断唯一性,应用服务负责验证,注意:是先验证,然后修改的实体。

1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6
7 using Happy.Command;
8 using Happy.Application;
9 using Happy.Example.Domain.TestGrids;
10 using Happy.Example.Commands.TestGrids;
11
12 namespace Happy.Example.Application.TestGrids
13 {
14 public class TestGridCommandHandler : ApplicationService,
15 ICommandHandler<CreateTestGridComamnd>,
16 ICommandHandler<UpdateTestGridComamnd>,
17 ICommandHandler<DeleteTestGridComamnd>
18 {
19 public void Handle(CreateTestGridComamnd command)
20 {
21 var testGridService = this.Service<TestGridService>();
22
23 testGridService.CheckNVarcharFieldUnique(command.NVarcharField);
24 var testGrid = command.CreateTestGrid();
25
26 testGridService.Create(testGrid);
27 command.Result = testGrid.Id;
28 }
29
30 public void Handle(UpdateTestGridComamnd command)
31 {
32 var testGridService = this.Service<TestGridService>();
33
34 var testGrid = testGridService.LoadAndDetach(command.Id);
35 if (testGrid.NVarcharField != command.NVarcharField)
36 {
37 testGridService.CheckNVarcharFieldUnique(command.NVarcharField);
38 }
39 command.UpdateTestGrid(testGrid);
40
41 testGridService.Update(testGrid);
42 }
43
44 public void Handle(DeleteTestGridComamnd command)
45 {
46 this.Service<TestGridService>().Delete(command.Id);
47 }
48 }
49 }


1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6
7 using AutoMapper;
8
9 using Happy.Infrastructure;
10 using Happy.Infrastructure.AutoMapper;
11 using Happy.Command;
12 using Happy.Example.Domain.TestGrids;
13
14 namespace Happy.Example.Commands.TestGrids
15 {
16 public class UpdateTestGridComamnd : ICommand, IHasIdProperty<Guid>
17 {
18 public Guid Id { get; set; }
19 public System.Int64? BigIntField { get; set; }
20 public System.Boolean? BitField { get; set; }
21 public System.DateTime? DateField { get; set; }
22 public System.DateTime? DateTimeField { get; set; }
23 public System.Decimal? DecimalField { get; set; }
24 public System.Double? FloatField { get; set; }
25 public System.Int32? IntField { get; set; }
26 public System.Decimal? MoneyField { get; set; }
27 public System.Decimal? NumericField { get; set; }
28 public System.String NVarcharField { get; set; }
29 public System.Single? RealField { get; set; }
30 public System.TimeSpan? TimeField { get; set; }
31 public System.Byte[] TimestampField { get; set; }
32 public byte[] OptimisticKey { get; set; }
33
34 internal void UpdateTestGrid(TestGrid testGrid)
35 {
36 Mapper.Map(this, testGrid);
37 testGrid.SetNVarcharField(this.NVarcharField);
38 }
39
40 static UpdateTestGridComamnd()
41 {
42 var map = Mapper.CreateMap<UpdateTestGridComamnd, TestGrid>();
43 map.ForMember(x => x.Id, m => m.Ignore());
44 map.IgnoreNotPublicSetter();
45 }
46 }
47 }


1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6
7 using Happy.Example.Domain.TestGrids;
8
9 namespace Happy.Example.Application.TestGrids
10 {
11 public partial class TestGridService
12 {
13 protected override void AfterCreate(TestGrid aggregate)
14 {
15 base.AfterCreate(aggregate);
16
17 aggregate.PublishCreatedEvent();
18 }
19
20 internal void CheckNVarcharFieldUnique(string value)
21 {
22 if (!this.Repository.IsNVarcharFieldExist(value))
23 {
24 throw new InvalidOperationException("NVarcharField必须唯一");
25 }
26 }
27 }
28 }

备注
一些好的资源:
- http://msdn.microsoft.com/en-us/library/ff664356(v=pandp.50).aspx
- http://gorodinski.com/blog/2012/05/19/validation-in-domain-driven-design-ddd/
- http://lostechies.com/jimmybogard/2009/02/15/validation-in-a-ddd-world/
- http://abdullin.com/journal/2009/1/29/ddd-and-rule-driven-ui-validation-in-net.html
- http://www.iteye.com/topic/413815
- http://devlicio.us/blogs/billy_mccafferty/archive/2009/02/17/a-response-to-validation-in-a-ddd-world.aspx
关于模型的合法性,Entity.IsValid()合理吗?的更多相关文章
- DDD:关于模型的合法性,Entity.IsValid()合理吗?
背景 见过很多框架(包括我自己的)都会在实体的定义中包含一个IsValid()方法,用来判断实体的合法性,是否应该这样设计呢?本文就这个问题介绍一点想法,希望大家多批评. 实体能否处于“非法”状态? ...
- Entity Framework:如果允许模型处于非法状态,在某些场景下,记得清空DbContext
Entity Framework:如果允许模型处于非法状态,在某些场景下,记得清空DbContext 背景 之前写过两篇文章介绍模型的合法性: DDD:关于模型的合法性,Entity.IsValid( ...
- ASP.NET MVC 5 - 添加一个模型
在本节中,您将添加一些类,这些类用于管理数据库中的电影.这些类是ASP.NET MVC 应用程序中的"模型(Model)". 您将使用.NET Framework 数据访问技术En ...
- Asp.Net MVC4入门指南(4):添加一个模型
在本节中,您将添加一些类,这些类用于管理数据库中的电影.这些类是ASP.NET MVC 应用程序中的"模型(Model)". 您将使用.NET Framework 数据访问技术En ...
- 自制C#版3DS文件的解析器并用SharpGL显示3DS模型
自制C#版3DS文件的解析器并用SharpGL显示3DS模型 我已经重写了3ds解析器,详情在此(http://www.cnblogs.com/bitzhuwei/p/CSharpGL-2-parse ...
- backbonejs中的模型篇(二)
一:模型标识符 每个模型都有一个用作唯一标识符的ID属性,以便在不同模型间进行区分.通过id属性我们可以直接访问模型对象当中用于标识符存放的属性,默认属性名为id,但也可以通过设置idAttribut ...
- Entity Framework 全面教程详解(转)
目录 预备知识 2 LINQ技术 2 LINQ技术的基础 - C#3.0 2 自动属性 2 隐式类型 2 对象初始化器与集合初始化器 3 匿名类 3 扩展方法 ...
- Getting Started with Entity Framework 6 Code First using MVC 5--Contoso 大学
在本教程中使用的软件版本 Visual Studio 2013 年 4.5.NET 实体框架 (EntityFramework 6.1.0 NuGet 包) 6 Windows Azure SDK 2 ...
- 转载Entity Framework全面教程
转载原地址:http://www.cnblogs.com/lsxqw2004/archive/2009/05/31/1495240.html#_Toc228672754 预备知识 2 LINQ技 ...
随机推荐
- ASP.NET MVC 插件化
ASP.NET MVC 插件化机制 2015-03-14 22:25 by 杨康新, 1328 阅读, 15 评论, 收藏, 编辑 概述 nopCommerce的插件机制的核心是使用BuildMana ...
- 枚举 UIButton补充
一.URL 1.什么是URL? URL是某个资源的唯一路径,通过这个路径就能访问对应的资源 2.URL的组成 协议头://全路径 * 协议头就代表资源的类型,比如http代表网络服务器资源,ftp代表 ...
- selenium之多线程启动grid分布式测试框架封装(四)
九.工具类,启动所有远程服务的浏览器 在utils包中创建java类:LaunchAllRemoteBrowsers package com.lingfeng.utils; import java.n ...
- C# 复习(1) 委托与事件
委托定义顺序 1. 声明一个委托 2.定义一个委托变量 3. 委托变量的初始化或者给委托变量绑定一个方法 4.调用委托 事件:事件是对委托的封装. 事件只能在创建事件的类的内部调用. public c ...
- 高德地图教程_poi搜索和显示
通过高仿深圳的应用近期打算.UI我们已经做了,我见过APP查询界面.关闭网络也将是能够查询其指示数据被存储在数据库中,或者是第一网络,所有网站上的数据是好了.我想简单地使用查询地图提供了. 曾经是接触 ...
- DDD分层架构之仓储
DDD分层架构之仓储(层超类型基础篇) 前一篇介绍了仓储的基本概念,并谈了我对仓储的一些认识,本文将实现仓储的基本功能. 仓储代表聚合在内存中的集合,所以仓储的接口需要模拟得像一个集合.仓储中有很多操 ...
- Maven学习笔记(一) : 简单介绍
近期在学习<maven实战>,写点东西记录一下.^_^ 何为Maven: Maven主要服务于基于java平台的项目构建.依赖管理和项目信息管理. Maven是优秀的构建工具: ...
- GIMP也疯狂之动态图的制作(二)
首先看下效果: (素材丢失,无法提供) 所用工具:GIMP.GIMP-GAP(在源中直接搜索安装) 文后会添加一个从U2B上搬运过来的视频教程,效果不错,值得一看本想也制作个人物变换,但几次实验,相同 ...
- 使用Visual Studio 2010 - 初学者系列 - 学习者系列文章
本文介绍Visual Studio 2010的基本使用. 1. 欢迎界面 2. 进入界面 3.选择菜单中的项目 4.选择项目路径,还有空白解决方案 5.选择 新建解决方案文件夹 6.选择新建项目 ...
- 设置符合条件的DataGridView的行的颜色的两种方法
Private Sub dgvInfo_RowPrePaint(sender As Object, e As DataGridViewRowPrePaintEventArgs) Handles dgv ...