12

首先,很抱歉问题很长,但我必须提供一些基本信息。

我们正在创建一个使用 ASP.net MVC、JQuery 模板、实体框架、WCF 的应用程序,并且我们使用 POCO 作为我们的域层。在我们的应用程序中,有一个 WCF 服务层与 ASP.net MVC 应用程序交换数据,它使用从 WCF 到 MVC 的数据传输对象 (DTO)。

此外,在我们的 WCF 服务层中转换 Domain-TO-DTO 时,应用程序通过使用 AutoMapper 在实体框架中使用延迟加载。

我们的后端架构如下(WCF Services -> Managers -> Repository -> Entity Framework(POCO)

在我们的应用程序中,我们不使用视图模型,因为我们不想要 MVC 应用程序的另一个映射层,我们只使用 DTO 作为视图模型。

通常,我们有针对客户、CustomerLite 等域的 Normal 和 Lite DTO(Lite 对象的属性比 Normal 少)。

现在我们在使用 DTO 时遇到了一些困难,因为我们的 DTO 结构变得越来越复杂,并且当我们考虑可维护性(使用 DTO 的一般层次结构)时,我们会失去性能。

例如,

我们有客户视图页面和我们的 DTO 层次结构如下

 public class CustomerViewDetailsDTO
 {
   public CustomerLiteDto Customer{get;set;}
   public OrderLiteDto Order{get;set;}
   public AddressLiteDto Address{get;set;}
 }

在这种情况下,我们不希望该视图的某些 OrderLiteDto 字段。但是其他一些视图需要该字段,因此为了方便它,我们使用该结构。

当涉及到自动映射时,我们映射 CustomerViewDetailsDTO,我们将从延迟加载(实体框架)中获取额外的数据(特定视图不需要)。

我的问题:

  1. 在考虑可维护性的同时,我们可以使用任何机制来提高性能吗?

  2. 是否可以将 Automapper 与更多基于地图视图的映射功能一起用于相同的 DTO ?

4

1 回答 1

11

首先不要使用延迟加载,因为它可能会导致 Select N+1 问题或类似问题。

Select N + 1 是一种数据访问反模式,其中以次优方式访问数据库。

换句话说,使用延迟加载而不预先加载集合会导致 Entity Framework 进入数据库并一次将结果带回一行。

将 jsRender 用于模板,因为它比 JQuery 模板快得多: Render Bemchmark ,这里有一个关于如何使用它的好信息:Reducing JavaScript Code Using jsRender Templates in HTML5 Applications

通常,我们有针对客户、CustomerLite 等域的 Normal 和 Lite DTO(Lite 对象的属性比 Normal 少)。

您的正常 DTO 可能是 ViewModel,因为 ViewModel 可能会或可能不会一对一映射到 DTO,并且 ViewModel 通常包含从视图推回的逻辑,或者帮助根据用户的响应将数据推回模型。DTO 没有任何行为,其目的是减少应用程序层之间的调用次数。

在考虑可维护性的同时,我们可以使用任何机制来提高性能吗?

为一个视图使用一个 ViewModel,您不必担心可维护性。就个人而言,我通常创建一个作为基础的抽象类,对于编辑、创建或列表,我继承该类并添加特定于视图的属性。因此,例如,创建视图不需要 PropertyId(因为有人可以劫持您的帖子并发布它),因此只有 Edit 和 List ViewModels 暴露了 PropertyId 属性。

是否可以将 Automapper 与更多基于地图视图的映射功能一起用于相同的 DTO ?

您可以使用 AutoMapper 来定义每张地图,但问题是,地图会有多复杂。每个视图使用一个 ViewModel,您的地图将易于编写和维护。我必须指出,不建议在数据访问代码中使用 Automapper,因为:

AutoMapper 的一个缺点是来自域对象的投影仍然强制查询和加载整个域对象。

来源:自动投影 LINQ 查询

您可以使用一组目前受限的扩展来加速数据访问代码中的映射:停止在您的数据访问代码中使用 AutoMapper

问候

于 2012-06-06T06:49:50.240 回答