`

DDD:关于模型的合法性,Entity.IsValid()合理吗? - 幸福框架

    博客分类:
  • DDD
阅读更多
原帖地址:http://www.cnblogs.com/happyframework/p/3158338.html

背景

见过很多框架(包括我自己的)都会在实体的定义中包含一个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://www.cnblogs.com/happyframework/p/3158338.html,转载请注明。

分享到:
评论

相关推荐

    org.apache.http.httpentity jar包-系列jar包

    import org.apache.http.client.entity.UrlEncodedFormEntity; import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPost; import org.apache.http.conn.scheme.Scheme; ...

    ejb3-persistence(找不到javax.persistence.Entity问题解决)

    “Caused by: java.lang.ClassNotFoundException: javax.persistence.Entity”,查找资料发现是缺少jar包. 添加此包就OK了。 更多说明:https://blog.csdn.net/kingmax54212008/article/details/82762624

    commons-pool-1.3.jar 和commons-dbcp-1.2.2.jar

    然而,当使用注解驱动的Hibernate(比如@Entity、@Repository等)时,可能会与基于HibernateDaoSupport的BaseDAO类发生冲突,因为两者都提供了对SessionFactory的管理方式。 在这种情况下,引入"commons-pool-1.3....

    org.apache.http.entity.mime

    在Android开发中,由于HttpClient库的稳定性和强大的功能,即使在Android API级别33及更高版本中不再内置,许多开发者依然选择使用Apache HttpClient,包括`org.apache.http.entity.mime`包中的类,来进行网络通信和...

    Packtpub.WCF.4.5.Multi-Layer.Services.Development.with.Entity.Framework.Dec.2012

    标题 "Packtpub.WCF.4.5.Multi-Layer.Services.Development.with.Entity.Framework.Dec.2012" 暗示这是一部关于使用Windows Communication Foundation(WCF)4.5开发多层服务,并结合Entity Framework的教程或书籍,...

    jersey所有的jar包

    18. **jersey-entity-filtering**: 提供了实体过滤器,可以控制返回给客户端的资源数据。 19. **jersey-extension**: 可能包含一些扩展功能或实验性特性。 20. **jersey-logging**: 提供了日志相关的支持,可以...

    Packt.WCF.4.5.Multi-Layer.Services.Development.with.Entity.Framework.Dec.2012

    ### WCF 4.5与Entity Framework:多层服务开发详解 #### 一、WCF 4.5简介与核心技术 Windows Communication Foundation (WCF) 是一个由微软提供的用于构建服务导向的应用程序(SOA)的框架。它为创建安全且可互...

    MySql.Data.Entity.6.10.9 + MySql.Data.6.10.9

    MySQL.Data.Entity.6.10.9 和 MySQL.Data.6.10.9 是两个针对MySQL数据库操作的重要组件,主要用于.NET Framework环境中的Entity Framework(EF)集成。在本篇文章中,我们将深入探讨这两个库以及它们如何协同工作,...

    CRM 定制开发基础.docx

    - 设置方式:`entity.Attributes["description"] = "This is a test.";` #### 三、Attribute常见类型作为查询条件 查询表达式(QueryExpression)是CRM系统中最常用的查询工具之一,它支持各种复杂的查询逻辑。 ...

    EFCore.Repository案例演示

    本案例是微软EntityFrameworkCore的一个仓储模式实现,这个仓储库不是我写的,而是使用了一个老外写的EntityFrameworkCore.Data.Repository。 IDE环境:VS2022 开发框架:ASP.NET Core6.0 WebApi ORM框架:Entity...

    spring_MVC源码

    弃用了struts,用spring mvc框架做了几个项目,感觉都不错,而且使用了注解方式,可以省掉一大堆配置文件。本文主要介绍使用注解方式配置的spring mvc,之前写的spring3.0 mvc和rest小例子没有介绍到数据层的内容,...

    Z.EntityFramework.Extensions破解 注册机

    Z.EntityFramework.Extensions 破解 注册机,详细破解方法

    Z.EntityFramework.Extensions注册机

    在IT行业中,Entity Framework(EF)是.NET框架下的一款非常流行的对象关系映射(ORM)工具,它允许开发者使用面向对象的编程方式来操作数据库,而无需关注底层的SQL语句。Z.EntityFramework.Extensions是针对EF的一...

    Devart.Entity.Developer.5.7.311,最新破解版

    Entity Developer is a powerful ORM designer for ADO.NET Entity Framework, NHibernate, LinqConnect, and LINQ to SQL. You can use Model-First and Database-First approaches to design your ORM model and ...

    Orleans.Providers.EntityFramework:Orleans谷物存储的Entity Framework Core实现。 更多供应商以后再来

    Orleans.Providers.EntityFramework Orleans谷物存储的Entity Framework Core实现。 有一些不错的功能缺失。 我不是特别需要它们,但是如果您有建议或想要帮助,将不胜感激。用法Nuget: ://...

    System.Data.Entity

    System.Data.Entity是.NET框架中一个关键的部分,它构成了Entity Framework的核心,这是一个强大的对象关系映射(ORM)框架,用于简化数据库操作。ORM允许开发人员使用面向对象的编程方式来处理数据库,从而避免了...

    SSD7 选择题。Multiple-Choice

    In the Entity-Relationship model, properties that characterize entities and relationships are modeled as (a) attributes (b) participation constraints (c) entity types (d) weak entities Correct...

    Python库 | django-entity-1.16.0.tar.gz

    资源分类:Python库 所属语言:Python 资源全名:django-entity-1.16.0.tar.gz 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059

Global site tag (gtag.js) - Google Analytics