本節將演示一些範例用例並討論可選場景。
從實體/聚合根類建立物件是實體生命週期的第一步。聚合/聚合根規則和最佳實踐部分 建議為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 沒有關於實體建立的業務規則,除了在建構函式中進行一些形式的驗證。但是,在某些情況下,實體建立應該檢查一些額外的業務規則
例如,假設您不希望在完全相同的標題已經存在問題的情況下建立問題。在哪裡實現這個規則? 在 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
方法檢查相同標題是否已經存在問題,並在這種情況下丟擲業務異常為了使用上述方法,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 不把問題儲存到資料庫中?」 我們認為這是應用服務的責任
因為,在儲存問題物件之前,應用程式服務可能需要對其進行額外的更改/操作。如果領域服務儲存它,則儲存操作將重複
當你檢查 IssueAppService 時,你會看到在 IssueManager.CreateAsync 中不儲存 Issue 到資料庫的好處。否則,我們將需要執行一次插入(在 IssueManager 中)和一次更新(在分配問題之後)
我們可以簡單地說 「因為它是一個核心領域邏輯,應該在領域層中實現」。然而,這帶來了一個新的問題: 「您如何判斷它是核心領域邏輯,而不是應用程式邏輯?」 (稍後我們將詳細討論其中的差異)
對於這個例子,一個簡單的問題可以幫助我們做出決定: 「如果我們有另一種方法(用例)來建立一個問題,我們是否仍然應用相同的規則?」 你可能會想 「為什麼我們有第二種製造問題的方式?」 然而,在現實生活中,你有:
綜上所述,不同的應用程式始終遵循這樣的規則:新問題的標題不能與任何現有問題的標題相同!他們與應用層無關! 這就是為什麼該邏輯是核心領域邏輯,應該位於領域層中,而不應該在應用程式服務中實現為重複的程式碼。