1

我们一直致力于为分层应用程序开发架构。我们计划有一个 ASP.NET MVC 表示层,它同时为移动浏览器和“普通”浏览器提供服务。也就是说,如果用户在移动设备上导航到网站,我们需要相同的 ASP.NET MVC 表示层来提供针对移动浏览器优化的视图,同时提供针对移动浏览器优化的“常规”视图如果用户使用的是台式机/笔记本电脑,则使用台式机/笔记本电脑浏览器。目前,我们不打算支持原生移动应用程序,但我们可能会在未来添加该功能。此外,该体系结构将具有域模型层——域实体、域服务等、应用程序服务、

但是,我一直在尝试找出域服务、应用程序服务和域实体中的行为(方法)之间的区别,因为它适用于我们正在开发的系统。下面是我们系统中的一个高级用例以供检查。

我们有报告和报告组。这两个实体之间存在多对多的关系。数据库表是:Reports <-> Jct_Reports_ReportGroups <-> ReportGroups,域实体看起来像:

Report class :
    public string ReportName { get; set; }
    ...
    public virtual ICollection<ReportGroup> ReportGroups { get; set; }

ReportGroup class :
    public string GroupName { get; set; }
    ...
    public virtual ICollection<Report> Reports { get; set; }

当然,用户可以创建一个报告(不创建报告组),他们可以创建一个包含一个或多个报告的报告组。此外,用户可以删除报告。当这种情况发生时,会发生很多事情。首先,我们检查报告是否在任何报告组中(单个报告可以在许多不同的报告组中)。如果在任何报告组中找到该报告,我们将遍历每个报告组并从报告组中删除该报告(通过从 Jct_Reports_ReportGroups 表中删除一行)。然后,对于包含要删除的报告的每个报告组,我们检查报告组中是否还有其他报告。如果报告组中没有其他报告,我们也会删除报告组(通过从 ReportGroups 表中删除行)。如果报告组中还有其他报告,我们不会删除该报告组。如果所有这些操作都成功,我们将删除用户选择的要删除的报告(通过删除 Reports 表中的行)。最后,向用户显示一条消息,说明报表删除是否成功。

我希望这是一个足够的用例来帮助理解我们整个应用程序的一小部分。

我在某处读到域服务封装了不适合域对象的业务逻辑,并且不是典型的 CRUD 操作,而外部消费者使用应用程序服务与您的系统对话——如果消费者需要访问 CRUD 操作,他们会暴露在这里。但是,我只是不了解 POCO 域实体中会使用哪些方法(业务逻辑),哪些业务逻辑将被视为域服务并包含在业务层中,以及哪些业务逻辑将被视为包含在应用服务层。所以,我的主要问题是:鉴于上述用例,关于如何在域实体中区分域服务、应用程序服务和行为(方法)的建议是什么?

4

1 回答 1

4

这是一个相当大的问题。

我更喜欢以下非严格的指导方针:

应用服务:假设我们有一组描述用户与系统交互的用例。假设我们有特定输入和输出的案例LoginCash transfer他们还描述了成功和失败的场景。有了这个,我们通常有一个应用程序服务规范:我会公开两个方法LoginResul Login(name, pass)TransferResult CashTrasnfer(from, to, amount)或多或少相同的输入和输出以及成功/失败行为。

很明显,为了实现用例,应用程序服务调用 BL(但添加了安全性和其他特定于应用程序的检查)。

域服务:正如您所描述的:“不自然地适合域对象”。

有了 CashTransnfer 用例,我们必须:

  • 加载帐户“来自”
  • 加载帐户“到”
  • check 'from' balance' //可能会返回错误
  • 验证帐户访问(锁定等)//可能返回错误
  • 从“来自”中提取“金额”
  • 把金额放在“到”
  • (理想情况下,这应该是一个单一的“业务”交易)。

很明显,这个函数不适合 Account 实体——所以最好有一个特殊TransferService的。


特别是关于报告/报告组功能 - 它可能是实体范围,因为没有非逻辑引用(所有项目都可以从报告中访问):

Report.Remove() {
    foreach(group in Groups) {
        group.RemoveReport(this);
    }
    repository.Remove(this);
} 
Group.RemoveReport(report) {
    reports.Remove(report);
    if(reports.Count == 0)
        repository.Remove(this);
}

或在报告服务和实体之间拆分(引用存储库可能有问题):

Report.RemoveFromAllGroups() {
   foreach(group in Groups) {
        group.RemoveReport(this);
        if(group.IsEmpty)
            //add to collection to return
    }
}
Service.Remove(report) 
{
    var emptyGroups = report.RemoveFromAllGroups();
    reportRepo.Remove(report);
    groupRepo.Remove(emptyGroups);
}
于 2012-05-31T10:14:00.050 回答