我正在(重新)设计大型应用程序,我们使用基于DDD的多层架构。

我们有带有数据层(存储库的实现)、领域层(域模型和接口的定义——存储库、服务、工作单元)、服务层(服务的实现)的MVC。到目前为止,我们在所有层上使用域模型(主要是实体),并且仅将dto用作视图模型(在控制器中,服务返回域模型,控制器创建视图模型,并将其传递给视图)。

I'v read countless articles about using, not using, mapping and passing DTOs. I understand that there's no any definitive answer, but I'm not sure if it's ok or not returning domain models from services to controllers. If I return domain model, it's still never passed to the view, since controller always creates view-specific view model - in this case, it seem legit. On the other hand, it doesn't feel right when domain model leaves business layer (service layer). Sometimes service needs to return data object that wasn't defined in the domain and then we either have to add new object to the domain that isn't mapped, or create POCO object (this is ugly, since some services return domain models, some effectively return DTOs).

The question is - if we strictly use view models, is it ok to return domain models all the way to controllers, or should we always use DTOs for communication with service layer? If so, is it ok to adjust domain models based on what services need? (Frankly I don't think so, since services should consume what domain has.) If we should strictly stick to DTOs, should they be defined in service layer? (I think so.) Sometimes it's clear that we should use DTOs (e.g., when service performs lot of business logic and creates new objects), sometimes it's clear that we should use just domain models (e.g., when Membership service returns anemic User(s) - it seems it wouldn't make much sense to create DTO that is the same as domain model) - but I prefer consistency and good practices.

文章域vs DTO vs ViewModel -如何以及何时使用它们?(以及其他一些文章)与我的问题非常相似,但它没有回答这个问题。我应该用EF在存储库模式中实现dto吗?也类似,但它不处理DDD。

免责声明:我不打算仅仅因为任何设计模式的存在而使用它,另一方面,我想使用好的设计模式和实践,因为它有助于设计应用程序作为一个整体,有助于分离关注点,即使使用特定的模式不是“必要的”,至少目前是这样。


当前回答

In my experience, unless you are using an OO UI pattern (like naked objects), exposing the domain objects to the UI is a bad idea. This because as the application grows, the needs from the UI change and force your objects to accommodate those changes. You end up serving 2 masters: UI and DOMAIN which is a very painful experience. Believe me, you don't want to be there. The UI model has the function of communicating with the user, the DOMAIN model to hold the business rules and the persistence models deals with storing data effectively. They all address different needs of the application. I'm in the middle of writing a blog post about this, will add it when it's done.

其他回答

我建议分析这两个问题:

Are your upper layers (i.e. view & view models / controllers) consuming the data in a different way of what the domain layer exposes? If there is a lot of mapping being done or even logic involved I'll suggest revisiting your design: it should probably be closer to how the data is actually used. How likely is it that you deeply change your upper layers? (e.g. swapping ASP.NET for WPF). If this is highly unlike and your architecture is not very complex, you may be better off exposing as many domain entities as you can.

恐怕这是一个相当广泛的话题,它实际上涉及到您的系统有多复杂及其需求。

In my experience, unless you are using an OO UI pattern (like naked objects), exposing the domain objects to the UI is a bad idea. This because as the application grows, the needs from the UI change and force your objects to accommodate those changes. You end up serving 2 masters: UI and DOMAIN which is a very painful experience. Believe me, you don't want to be there. The UI model has the function of communicating with the user, the DOMAIN model to hold the business rules and the persistence models deals with storing data effectively. They all address different needs of the application. I'm in the middle of writing a blog post about this, will add it when it's done.

当域模型离开业务层(服务层)时,感觉不对

让你觉得你在把内脏掏出来,对吧?根据Martin Fowler的说法:服务层定义了应用程序的边界,它封装了域。换句话说,它保护了定义域。

有时服务需要返回域中未定义的数据对象

您能否提供此数据对象的示例?

如果我们应该严格遵循dto,那么它们应该在服务层定义吗?

是的,因为响应是服务层的一部分。如果它被定义为“其他地方”,那么服务层需要引用该“其他地方”,从而为您的千层面添加一个新层。

是否可以将域模型一直返回到控制器,或者我们应该始终使用dto与服务层通信?

DTO是一个响应/请求对象,如果您使用它进行通信,那么它是有意义的。如果你在表示层(MVC-Controllers/View, WebForms, ConsoleApp)中使用域模型,那么表示层与你的域紧密耦合,域中的任何变化都需要你改变控制器。

创建与域模型相同的DTO似乎没有多大意义)

这是DTO的缺点之一。现在,您正在考虑复制代码,但随着项目的扩展,它将变得更有意义,特别是在不同团队被分配到不同层的团队环境中。

DTO可能会给应用程序增加额外的复杂性,但层也是如此。DTO是系统的一个昂贵功能,它们不是免费的。

为什么使用DTO

本文提供了使用DTO http://guntherpopp.blogspot.com/2010/09/to-dto-or-not-to-dto.html的优点和缺点

总结如下:

何时使用

用于大型项目。 项目寿命10年以上。 战略,关键任务应用。 大型团队(5人以上) 开发人员在地理上分布。 域和表示是不同的。 减少开销数据交换(DTO的最初目的)

何时不使用

小型至中型项目(最多5人) 项目生命周期为2年左右。 GUI、后台等没有单独的团队。

反对DTO的理由

代码复制。 开发时间成本,调试成本。(使用DTO生成工具http://entitiestodtos.codeplex.com/) 您必须始终同步这两个模型。(就我个人而言,我喜欢这个,因为它有助于了解变化的涟漪效应) 开发成本:额外的映射是必要的。(使用自动绘图器,如https://github.com/AutoMapper/AutoMapper) 为什么数据传输对象(dto)是反模式?

DTO参数

如果没有DTO,表示和域是紧密耦合的。(这适用于小型项目。) 接口/ API稳定 可以通过返回只包含绝对需要的属性的DTO来为表示层提供优化。使用linq投影,你不需要拉出整个实体。 为了降低开发成本,请使用代码生成工具

根据我的经验,你应该做些实际的事情。“最好的设计是最简单的设计”——爱因斯坦。有了这样的思想……

如果我们严格使用视图模型,是否可以将域模型一直返回到控制器,或者我们应该始终使用dto与服务层通信?

当然没关系!如果你有域实体,DTO和视图模型,那么包括数据库表,你在应用程序中重复的所有字段在4个地方。我曾参与过一些大型项目,其中领域实体和视图模型工作得很好。唯一的例外是,如果应用程序是分布式的,而服务层位于另一台服务器上,在这种情况下,由于序列化的原因,需要通过网络发送dto。

如果是,是否可以根据服务需求调整域模型?(坦白地说,我不这么认为,因为服务应该消费域拥有的东西。)

一般来说,我同意并说不,因为域模型通常是业务逻辑的反映,通常不是由该逻辑的消费者塑造的。

如果我们应该严格遵循dto,那么它们应该在服务层定义吗?(我想是的。)

如果你决定使用它们,我同意并说是的,服务层是一个完美的地方,因为它在一天结束时返回dto。

好运!

到目前为止,我们在所有层上使用域模型(主要是实体),并且仅将dto用作视图模型(在控制器中,服务返回域模型,控制器创建视图模型,并将其传递给视图)。

由于领域模型为整个应用程序提供了术语(泛在语言),因此最好广泛使用领域模型。

使用ViewModels/ dto的唯一原因是在应用程序中实现MVC模式,以分离视图(任何类型的表示层)和模型(域模型)。在这种情况下,您的表示和域模型是松散耦合的。

有时服务需要返回域中未定义的数据对象,然后我们必须向未映射的域添加新对象,或者创建POCO对象(这很难看,因为有些服务返回域模型,有些则有效地返回dto)。

我假设您谈论的是应用程序/业务/域逻辑服务。

我建议您尽可能地返回域实体。如果需要返回额外的信息,返回包含多个域实体的DTO是可以接受的。

有时,使用第三方框架的人,在域实体上生成代理,会面临从他们的服务中暴露域实体的困难,但这只是错误使用的问题。

问题是——如果我们严格使用视图模型,是否可以将域模型一直返回到控制器,或者我们应该始终使用dto与服务层通信?

我想说在99.9%的情况下返回域实体就足够了。

为了简化dto的创建并将域实体映射到dto中,您可以使用AutoMapper。