用例演示 - 创建实体

本节将演示一些示例用例并讨论可选场景。

创建实体

从实体/聚合根类创建对象是实体生命周期的第一步。聚合/聚合根规则和最佳实践部分建议为Entity类创建一个主构造函数,以保证创建一个有效的实体。因此,无论何时我们需要创建实体的实例,我们都应该使用那个构造函数

参见下面的问题聚合根类:

public class Issue : AggregateRoot<Guid>
{
public Guid RepositoryId { get; private set; }
public string Title { get; private set; }
public string Text { get; set; }
public Guid? AssignedUserId { get; private set; } public Issue(
Guid id,
Guid repositoryId,
string title,
string text = null
) : base(id)
{
RepositoryId = repositoryId;
Title = Check.NotNullOrWhiteSpace(title, nameof(title));
Text = text; // 允许空值
} private Issue() { //为ORM保留的空构造函数 } public void SetTitle(string title)
{
Title = Check.NotNullOrWhiteSpace(title, nameof(title));
}
}
  • 该类保证通过其构造函数创建有效的实体

  • 如果你需要更改标题,你需要使用 SetTitle 方法保证标题在一个有效状态

  • 如果您想将这个问题分配给用户,您需要使用 IssueManager (它在分配之前实现了一些业务规则, 请参阅我之前关于 领域服务 的文章)。

  • Text 属性有一个公共setter,因为它也接受null值,并且这个示例没有任何验证规则。它在构造函数中也是可选的

让我们看看用于创建问题的Application Service方法:

public class IssueAppService : ApplicationService, IIssueAppService
{
//省略了Repository和DomainService的依赖注入 [Authorize]
public async Task<IssueDto> CreateAsync(IssueCreationDto input)
{
//创建一个有效的问题实体
var issue = new Issue(
GuidGenerator.Create(),
input.RepositoryId,
input.Title,
input.Text
); //如果传入了被分配人,则把该问题法分配给这个用户
if(input.AssignedUserId.HasValue)
{
var user = await _userRepository.GetAsync(input.AssignedUserId.Value);
await _issueManager.AssignToAsync(issue, user);
} // 把问题实体保存到数据库
await _issueRepository.InsertAsync(issue); //返回表示这个新的问题的DTO
return ObjectMapper.Map<Issue, IssueDto>(issue);
}
}

CreateAsync 方法:

  • 使用 Issue 构造函数创建有效的问题。它使用 IGuidGenerator 服务传递Id。这里不使用自动对象映射
  • 如果客户端希望在对象创建时将这个问题分配给用户,它会使用IssueManager 来完成,允许 IssueManager 在分配之前执行必要的检查。
  • 保存实体到数据库
  • 最后使用 IObjectMapper 返回一个 IssueDto ,该 IssueDto 是通过映射从新的 Issue 实体自动创建的

使用领域规则创建实体

上述示例, Issue 没有关于实体创建的业务规则,除了在构造函数中进行一些形式的验证。但是,在某些情况下,实体创建应该检查一些额外的业务规则

例如,假设您不希望在完全相同的标题已经存在问题的情况下创建问题。在哪里实现这个规则? 在 Application Service 中实现此规则是不合适的,因为它是一个应该始终检查的 核心业务(领域)规则

该规则应该在 领域服务 (在本例中是 IssueManager )中实现。因此,我们需要强制应用层总是使用 IssueManager 来创建一个新的 Issue

首先,我们可以将 Issue 构造函数设置为 internal ,而不是 public:

public class Issue : AggregateRoot<Guid>
{
internal Issue(
Guid id,
Guid repositoryId,
string title,
string text = null
) : base(id)
{
//...
}
}

这阻止了应用服务直接使用构造函数,所以它们将使用 IssueManager 。然后我们可以在 IssueManager 中添加一个 CreateAsync 方法:

public class IssueManager : DomainService
{
//省略了依赖注入 public async Task<IssueDto> CreateAsync(
Guid repositoryId,
string title,
string text = null
)
{
//如果存在相同标题的问题,直接抛错
if(await _issueRepository.AnyAsync(i => i.Title == title))
{
throw new BusinessException("IssueTracking:IssueWithSameTitleExists");
} //创建一个有效的问题实体
return new Issue(
GuidGenerator.Create(),
repositoryId,
title,
text
);
}
}
  • CreateAsync 方法检查相同标题是否已经存在问题,并在这种情况下抛出业务异常
  • 如果没有重复,则创建并返回一个新的Issue

为了使用上述方法,IssueAppService 被修改如下:

public class IssueAppService : ApplicationService, IIssueAppService
{
//省略了依赖注入 public async Task<IssueDto> CreateAsync(IssueCreationDto input)
{
//★修改为通过领域服务创建有效的问题实体, 而不是直接new
var issue = await _issueManager.CreateAsync(
GuidGenerator.Create(),
input.RepositoryId,
input.Title,
input.Text
); //如果传入了被分配人,则把该问题法分配给这个用户
if(input.AssignedUserId.HasValue)
{
var user = await _userRepository.GetAsync(input.AssignedUserId.Value);
await _issueManager.AssignToAsync(issue, user);
} // 把问题实体保存到数据库
await _issueRepository.InsertAsync(issue); //返回表示这个新的问题的DTO
return ObjectMapper.Map<Issue, IssueDto>(issue);
}
}

讨论:为什么问题没有在 IssueManager 中保存到数据库?

你可能会问 “为什么 IssueManager 不把问题保存到数据库中?” 我们认为这是应用服务的责任

因为,在保存问题对象之前,应用程序服务可能需要对其进行额外的更改/操作。如果领域服务保存它,则保存操作将重复

  • 两次数据库往返会导致性能损失
  • 需要显式的数据库事务来包含这两个操作
  • 如果由于业务规则的原因,其他操作取消了实体创建,则应该在数据库中回滚事务

当你检查 IssueAppService 时,你会看到在 IssueManager.CreateAsync 中不保存 Issue 到数据库的好处。否则,我们将需要执行一次插入(在 IssueManager 中)和一次更新(在分配问题之后)

讨论:为什么不在应用程序服务中实现重复标题检查?

我们可以简单地说 “因为它是一个核心领域逻辑,应该在领域层中实现”。然而,这带来了一个新的问题: “您如何判断它是核心领域逻辑,而不是应用程序逻辑?” (稍后我们将详细讨论其中的差异)

对于这个例子,一个简单的问题可以帮助我们做出决定: “如果我们有另一种方法(用例)来创建一个问题,我们是否仍然应用相同的规则?” 你可能会想 “为什么我们有第二种制造问题的方式?” 然而,在现实生活中,你有:

  • 应用程序的最终用户可能会在应用程序的标准UI中创建问题(比如在github的网页端创建问题)
  • 您可能有第二个后台应用程序,由您自己的员工使用,您可能希望提供一种创建问题的方法(在本例中可能使用不同的授权规则)
  • 您可能有一个对第三方客户端开放的HTTP API,他们会创建问题。
  • 您可能有一个 background worker service,如果它检测到一些故障,它会做一些事情并创建问题。这样,它将在没有任何用户交互的情况下(可能没有任何标准的授权检查)创建问题。
  • 您甚至可以在UI上设置一个按钮,将某些内容 (例如,讨论) 转换为问题

综上所述,不同的应用程序始终遵循这样的规则:新问题的标题不能与任何现有问题的标题相同!他们与应用层无关! 这就是为什么该逻辑是核心领域逻辑,应该位于领域层中,而不应该在应用程序服务中实现为重复的代码。

实现领域驱动设计 - 使用ABP框架 - 创建实体的更多相关文章

  1. 实现领域驱动设计 - 使用ABP框架 - 什么是领域驱动设计?

    前言: 最近看到ABP官网的一本电子书,感觉写的很好,翻译出来,一起学习下 (Implementing Domain Driven Design) https://abp.io/books DDD简介 ...

  2. 实现领域驱动设计 - 使用ABP框架 - 通用准则

    在进入细节之前,让我们看看一些总体的 DDD 原则 数据库提供者 / ORM 无关性 领域和应用程序层应该与 ORM / 数据库提供程序 无关.它们应该只依赖于 Repository 接口,而 Rep ...

  3. 实现领域驱动设计 - 使用ABP框架 - 解决方案概览

    .NET解决方案的分层 下图显示了使用ABP的 应用启动模板 创建的Visual Studio解决方案: 解决方案名称为问题跟踪,它由多个项目组成.通过考虑DDD原则以及开发和部署实践,该解决方案是分 ...

  4. 实现领域驱动设计 - 使用ABP框架 - 存储库

    存储库 Repository 是一个类似于集合的接口,领域层和应用程序层使用它来访问数据持久性系统(数据库),以读写业务对象(通常是聚合) 常见的存储库原则是: 在领域层定义一个存储库接口(因为它被用 ...

  5. .net core +codefirst(.net core 基础入门,适合这方面的小白阅读) 【我们一起写框架】领域驱动设计的CodeFirst框架(一)—序篇

    .net core +codefirst(.net core 基础入门,适合这方面的小白阅读)   前言 .net core mvc和 .net mvc开发很相似,比如 视图-模型-控制器结构.所以. ...

  6. 【我们一起写框架】领域驱动设计的CodeFirst框架(一)—序篇

    前言 领域驱动设计,其实已经是一个很古老的概念了,但它的复杂度依旧让学习的人头疼不已. 互联网关于领域驱动的文章有很多,每一篇写的都很好,理解领域驱动设计的人都看的懂. 不过,这些文章对于那些初学者而 ...

  7. DDD 领域驱动设计-“臆想”中的实体和值对象

    其他博文: DDD 领域驱动设计-三个问题思考实体和值对象 DDD 领域驱动设计-三个问题思考实体和值对象(续) 以下内容属于博主"臆想",如有不当,请别当真. 扯淡开始: 诺兰的 ...

  8. (转)EntityFramework之领域驱动设计实践

    EntityFramework之领域驱动设计实践 - 前言 EntityFramework之领域驱动设计实践 (一):从DataTable到EntityObject EntityFramework之领 ...

  9. EntityFramework之领域驱动设计实践

    EntityFramework之领域驱动设计实践 - 前言 EntityFramework之领域驱动设计实践 (一):从DataTable到EntityObject EntityFramework之领 ...

随机推荐

  1. switch 用法

    1.语法格式和规则 switch case 语句语法格式如下: switch(expression){ case value : //语句 break; //可选 case value : //语句 ...

  2. 论文翻译:2018_LSTM剪枝_Learning intrinsic sparse structures within long short-term memory

    论文地址:在长短时记忆中学习内在的稀疏结构 论文代码:https://github.com/wenwei202/iss-rnns 引用格式:Wen W, He Y, Rajbhandari S, et ...

  3. 论文解读(BGRL)《Bootstrapped Representation Learning on Graphs》

    论文信息 论文标题:Bootstrapped Representation Learning on Graphs论文作者:Shantanu Thakoor, Corentin Tallec, Moha ...

  4. JS_Window-三种消息框:警告框、确认框、提示框、页面显示时间-计时-延时

    1 <!DOCTYPE html> 2 <html lang="en"> 3 <head> 4 <meta charset="U ...

  5. petite-vue源码剖析-沙箱模型

    在解析v-if和v-for等指令时我们会看到通过evaluate执行指令值中的JavaScript表达式,而且能够读取当前作用域上的属性.而evaluate的实现如下: const evalCache ...

  6. springmvc03-restful和控制器

    一.控制器Controller 控制器复杂提供访问应用程序的行为,通常通过接口定义或注解定义两种方法实现. 控制器负责解析用户的请求并将其转换为一个模型. 在Spring MVC中一个控制器类可以包含 ...

  7. 封闭的一个多月,老菜鸟的 机械手和AGV 自动搬运小项目总结

    最近上海疫情严重,闲赋在家无事可做,手机里不断的推送一些无脑的谩骂声音,索性找点事情做,将3月份实施的一个自动搬运小项目做一个简单的汇总,便于今后项目实施中积累一些经验.项目需求非常简单,因为能力有限 ...

  8. 使用CreateThreadPool创建线程池

    使用Windows API函数来创建线程池,可以极大的方便了自己编写线程池的繁琐步骤. 使用CreateThreadPool来创建一个线程池,需要在创建完成后,初始化线程池的状态,并且在不需要的时候清 ...

  9. CSS躬行记(11)——管理后台响应式改造

    为了提升业务人员操作管理后台的体验,花了点时间进行响应式的改造,紧急情况时,掏出手机就能工作. 利用CSS3的媒体查询,就能根据不同屏幕的尺寸采用不同的样式来渲染,目前使用的移动端屏幕阈值为750px ...

  10. C#开发PACS医学影像三维重建(十三):基于人体CT值从皮肤渐变到骨骼的梯度透明思路

    当我们将CT切片重建为三维体之后,通常会消除一些不必要的外部组织来观察内部病灶, 一般思路是根据人体常见CT值范围来使得部分组织透明来达到效果, 但这是非黑即白的,即,要么显示皮肤,要么显示神经,要么 ...