DDD:关于模型的合法性,Entity.IsValid()合理吗?
背景
见过很多框架(包括我自己的)都会在实体的定义中包含一个IsValid()方法,用来判断实体的合法性,是否应该这样设计呢?本文就这个问题介绍一点想法,希望大家多批评。
实体能否处于“非法”状态?
实体是否应该包含IsValid()方法的深层次问题是:“实体能否处于非法状态?”。我们来定义一些术语,接下来我就引用这些术语:
- A模式:实体允许处于非法状态,但是实体要包含一个IsValid()方法进行校验。
- B模式:实体不允许处于非法状态,业务逻辑必须保证这一点。
关于A模式我不想多说了,A模式本身没有问题的,今天重点说说如何实现B模式。
如何实现B模式?
最好的说明就是写一个例子,下面是我们例子的需求:
- xxx属性不能为空。
- xxx属性必须唯一。
这个例子非常简单,也具有代表性,可以进一步抽象为:
- xxx属性不能为空,聚合自身的验证。
- xxx属性必须唯一,跨聚合验证。
让我们一个一个来。
xxx属性不能为空,聚合自身的验证。
聚合本身应该负责自己状态的完整性,反射可能会绕过这些验证,使用类似AutoMapper的工具需要注意(我已经处理了)。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; using Happy.Domain;
using Happy.Domain.Tree;
using Happy.Example.Events.TestGrids; using Happy.Infrastructure; namespace Happy.Example.Domain.TestGrids
{
public partial class TestGrid : AggregateRoot<Guid>
{
public System.Int64? BigIntField { get; set; }
public System.Boolean? BitField { get; set; }
public System.DateTime? DateField { get; set; }
public System.DateTime? DateTimeField { get; set; }
public System.Decimal? DecimalField { get; set; }
public System.Double? FloatField { get; set; }
public System.Int32? IntField { get; set; }
public System.Decimal? MoneyField { get; set; }
public System.Decimal? NumericField { get; set; }
public System.String NVarcharField { get; private set; }
public System.Single? RealField { get; set; }
public System.TimeSpan? TimeField { get; set; }
public System.Byte[] TimestampField { get; set; } public void SetNVarcharField(string value)
{
value.MustNotNullAndNotWhiteSpace(value); this.NVarcharField = value;
} internal void PublishCreatedEvent()
{
this.PublishEvent(new TestGridCreatedEvent());
}
}
}
xxx属性必须唯一,跨聚合验证。
仓储负责判断唯一性,应用服务负责验证,注意:是先验证,然后修改的实体。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; using Happy.Command;
using Happy.Application;
using Happy.Example.Domain.TestGrids;
using Happy.Example.Commands.TestGrids; namespace Happy.Example.Application.TestGrids
{
public class TestGridCommandHandler : ApplicationService,
ICommandHandler<CreateTestGridComamnd>,
ICommandHandler<UpdateTestGridComamnd>,
ICommandHandler<DeleteTestGridComamnd>
{
public void Handle(CreateTestGridComamnd command)
{
var testGridService = this.Service<TestGridService>(); testGridService.CheckNVarcharFieldUnique(command.NVarcharField);
var testGrid = command.CreateTestGrid(); testGridService.Create(testGrid);
command.Result = testGrid.Id;
} public void Handle(UpdateTestGridComamnd command)
{
var testGridService = this.Service<TestGridService>(); var testGrid = testGridService.LoadAndDetach(command.Id);
if (testGrid.NVarcharField != command.NVarcharField)
{
testGridService.CheckNVarcharFieldUnique(command.NVarcharField);
}
command.UpdateTestGrid(testGrid); testGridService.Update(testGrid);
} public void Handle(DeleteTestGridComamnd command)
{
this.Service<TestGridService>().Delete(command.Id);
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; using AutoMapper; using Happy.Infrastructure;
using Happy.Infrastructure.AutoMapper;
using Happy.Command;
using Happy.Example.Domain.TestGrids; namespace Happy.Example.Commands.TestGrids
{
public class UpdateTestGridComamnd : ICommand, IHasIdProperty<Guid>
{
public Guid Id { get; set; }
public System.Int64? BigIntField { get; set; }
public System.Boolean? BitField { get; set; }
public System.DateTime? DateField { get; set; }
public System.DateTime? DateTimeField { get; set; }
public System.Decimal? DecimalField { get; set; }
public System.Double? FloatField { get; set; }
public System.Int32? IntField { get; set; }
public System.Decimal? MoneyField { get; set; }
public System.Decimal? NumericField { get; set; }
public System.String NVarcharField { get; set; }
public System.Single? RealField { get; set; }
public System.TimeSpan? TimeField { get; set; }
public System.Byte[] TimestampField { get; set; }
public byte[] OptimisticKey { get; set; } internal void UpdateTestGrid(TestGrid testGrid)
{
Mapper.Map(this, testGrid);
testGrid.SetNVarcharField(this.NVarcharField);
} static UpdateTestGridComamnd()
{
var map = Mapper.CreateMap<UpdateTestGridComamnd, TestGrid>();
map.ForMember(x => x.Id, m => m.Ignore());
map.IgnoreNotPublicSetter();
}
}
}
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; using Happy.Example.Domain.TestGrids; namespace Happy.Example.Application.TestGrids
{
public partial class TestGridService
{
protected override void AfterCreate(TestGrid aggregate)
{
base.AfterCreate(aggregate); aggregate.PublishCreatedEvent();
} internal void CheckNVarcharFieldUnique(string value)
{
if (!this.Repository.IsNVarcharFieldExist(value))
{
throw new InvalidOperationException("NVarcharField必须唯一");
}
}
}
}
备注
一些好的资源:
- 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
DDD:关于模型的合法性,Entity.IsValid()合理吗?的更多相关文章
- 关于模型的合法性,Entity.IsValid()合理吗?
关于模型的合法性,Entity.IsValid()合理吗? 背景 见过很多框架(包括我自己的)都会在实体的定义中包含一个IsValid()方法,用来判断实体的合法性,是否应该这样设计呢?本文就这个问题 ...
- Entity Framework:如果允许模型处于非法状态,在某些场景下,记得清空DbContext
Entity Framework:如果允许模型处于非法状态,在某些场景下,记得清空DbContext 背景 之前写过两篇文章介绍模型的合法性: DDD:关于模型的合法性,Entity.IsValid( ...
- DDD 领域驱动设计-在动手之前,先把你的脑袋清理干净
惨不忍睹的翻译 英文原文:http://www.codeproject.com/Articles/339725/Domain-Driven-Design-Clear-Your-Concepts-Bef ...
- 从壹开始微服务 [ DDD ] 之八 ║剪不断理还乱的 值对象和Dto
缘起 哈喽大家周四好,时间是过的真快,这几天一直忙着在公司的项目,然后带带新人,眼看这周要过去了,还是要抽出时间学习学习,这些天看到群里的小伙伴也都在忙着新学习,还是很开心的,至少当时的初衷已经达到了 ...
- 基于DDD的微服务设计和开发实战
你是否还在为微服务应该拆多小而争论不休?到底如何才能设计出收放自如的微服务?怎样才能保证业务领域模型与代码模型的一致性?或许本文能帮你找到答案. 本文是基于 DDD 的微服务设计和开发实战篇,通过借鉴 ...
- 驱动领域DDD的微服务设计和开发实战
你是否还在为微服务应该拆多小而争论不休?到底如何才能设计出收放自如的微服务?怎样才能保证业务领域模型与代码模型的一致性?或许本文能帮你找到答案. 本文是基于 DDD 的微服务设计和开发实战篇,通过借鉴 ...
- Lind.DDD.LindAspects方法拦截的介绍
回到目录 什么是LindAspects 之前写了关于Aspects的文章<Lind.DDD.Aspects通过Plugins实现方法的动态拦截~Lind里的AOP>,今天主要在设计思想上进 ...
- ASP.NET MVC 5 - 添加一个模型
在本节中,您将添加一些类,这些类用于管理数据库中的电影.这些类是ASP.NET MVC 应用程序中的"模型(Model)". 您将使用.NET Framework 数据访问技术En ...
- 我也来说说DDD~大话目录
回到占占推荐博客索引 DDD之前没有接触过,但一但有了接触就一发不可收拾,他会带去进入一个全新的世界! DDD不是新技术,而是新思想,新模式,是软件开发领域的一次突破,它更接近于业务,对于业务的改动它 ...
随机推荐
- Mybatis多参传递的四种解决方案
Mybatis多参传递的四种解决方案 代码异常:org.apache.ibatis.binding.BindingException: Parameter 'param' not found. 长时间 ...
- MySQL数据库定义与操作语言
文章为作者原创,未经许可,禁止转载. -Sun Yat-sen University 冯兴伟 实验1.1 数据库定义 (1)实验目的 理解和掌握数据库DDL语言,能够熟练地使用SQL DDL语句 ...
- 炜煌T3POS58微打参数设置方法
本文用菊子曰发布
- 一步一步写miscdevice的驱动模块
(本文使用的平台为友善tiny210SDKv2) 对于linux的驱动程序来说,主要分为三种:miscdevice.platform_device.platform_driver . 这三个结构体关系 ...
- Android中Parcelable接口的使用
在做开发的过程中,序列化是非常常见的.比如要将对象保存本地磁盘或者在网络上传输等.实现序列化有两种方式,一种是实现Serializable接口,第二种是实现Parcelable. Serializab ...
- ElasticSearch + Kibana
关键词: 数据可视化 数据分析 数据爬虫 信息检索(搜索引擎) ElasticSearch是基于Lucene的分布式搜索引擎,提供多种插件及配套工具. 其中Kibana可以“关联”ES中的数据集,进行 ...
- 用git写书
apebook.org www.apebook.org 最好的程序员图书免费托管服务 apebook 提供了 gitbook.com 类似的云端图书托管能力,图书基于广受欢迎的 gitbook 工具 ...
- gradle介绍
Gradle是什么? https://gradle.org/whygradle-build-automation/ 自动化编译的工具,可编程,适合各种语言,管理各种依赖,高效,提供分析报告. 我个人觉 ...
- TargetProcess 中更改了域名如何修改里面的附件地址
在默认情况,TP 安装的目录是默认的网站(IIS)的targetprocess2下面,但是如果想采用 tp.targetprocess.cn 这种方式直接访问,则需要做一些设置. 基本要点如下: 1. ...
- fio terse输出详解
fio, the flexible IO tester, is a very useful tool for benchmarking IO performance. It has an option ...