實現領域驅動設計

2022-06-24 12:00:59

用例演示 - 建立實體

本節將演示一些範例用例並討論可選場景。

建立實體

從實體/聚合根類建立物件是實體生命週期的第一步。聚合/聚合根規則和最佳實踐部分 建議為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上設定一個按鈕,將某些內容 (例如,討論) 轉換為問題

綜上所述,不同的應用程式始終遵循這樣的規則:新問題的標題不能與任何現有問題的標題相同!他們與應用層無關! 這就是為什麼該邏輯是核心領域邏輯,應該位於領域層中,而不應該在應用程式服務中實現為重複的程式碼。