当你修改Domain中的字段必须进行验证判断的、必须查询数据库的(需要查询其他领域的数据的)这时候(之前的AbpVnext中的DDD指南之聚合根的文章)里实体的简单判断就不合适了
这时候就需要用到DomainService,
/// <summary>
/// 问题的领域服务
/// </summary>
public class IssueManager : DomainService
{
private readonly IRepository<Issue, Guid> _issueRepository;
public IssueManager(IRepository<Issue, Guid> issueRepository)
{
_issueRepository = issueRepository;
}
public async Task AssignToAsync(Issue issue, Guid userId)
{
//不允许用户有超过3个打开的问题
var openIssueCount = await _issueRepository.CountAsync(i => i.AssignedUserId == userId && !i.IsClosed);
if (openIssueCount >= 3)
{
throw new BusinessException(code: "123", message: "用户已经绑定超过3个已打开的问题");
}
issue.AssignedUserId = userId;
}
}
当我们需要在问题上绑定某个用户时(也就是给AssignedUserId赋值),必须要判断用户是否绑定了3个问题,我们不允许用户绑定三个打开的问题,这时候我们需要查询数据库,我们注入Issue仓储,查询该用户是否有>=3个打开的问题,有的话我们抛出异常,没有的话我们给AssignedUserId赋值
这里AssignToAsync 就是领域逻辑,和前面聚合根的相比,他需要查询数据库,所以我们用到了领域服务
这里看我们的方法很简单,目的只有一个,就是给AssignedUserId 赋值,所以为什么前面AssignedUserId 是Internal的set方法
AbpVnext建议领域服务以Manager后缀命名
很简单吧