18

在我的 ASP.NET MVC 应用程序中,我使用工作单元和存储库模式进行数据访问。

使用工作单元类和其中定义的存储库,我在控制器中获取相关的实体集。以我的初学者知识,我可以想到两种方法来获取业务模型并将其转换为视图模型。

  • 存储库将业务模型返回给控制器,该模型比映射到视图模型,或者
  • 存储库本身将业务模型转换为视图模型,然后返回给控制器。

目前我正在使用第一种方法,但是对于具有很多属性的视图模型,我的控制器代码开始变得丑陋和冗长。

另一方面,我在想,由于我的存储库被称为 UserRepository(例如),它应该直接返回业务模型,而不是某些仅对单个视图有用的模型。

您认为其中哪一个是大型项目的更好实践?有替代方法吗?

4

3 回答 3

22

存储库应该返回域模型,而不是视图模型。至于模型和视图模型之间的映射,我个人使用AutoMapper,所以我有一个单独的映射层,但是这个层是从控制器调用的。

以下是典型的 GET 控制器操作的样子:

public ActionResult Foo(int id)
{
    // the controller queries the repository to retrieve a domain model
    Bar domainModel = Repository.Get(id);

    // The controller converts the domain model to a view model
    // In this example I use AutoMapper, so the controller actually delegates
    // this mapping to AutoMapper but if you don't have a separate mapping layer
    // you could do the mapping here as well.
    BarViewModel viewModel = Mapper.Map<Bar, BarViewModel>(domainModel);

    // The controller passes a view model to the view
    return View(viewModel);
}

当然可以使用自定义操作过滤器来缩短它以避免重复的映射逻辑:

[AutoMap(typeof(Bar), typeof(BarViewModel))]
public ActionResult Foo(int id)
{
    Bar domainModel = Repository.Get(id);
    return View(domainModel);
}

AutoMap 自定义操作过滤器订阅 OnActionExecuted 事件,拦截传递给视图结果的模型,调用映射层(在我的例子中为 AutoMapper)将其转换为视图模型并将其替换为视图。视图当然是视图模型的强类型。

于 2012-07-13T12:10:06.577 回答
4

我认为您的存储库应该返回业务模型。

然后,您可以使用Automapper之类的工具将属性自动映射到您的视图模型,并且可以摆脱手动映射代码。如果您不想将所有业务实体的属性或其复杂结构暴露给视图,这种方法非常有用。

您也可能会发现这篇文章很有帮助,您可以在其中摆脱手动映射调用(在某种程度上),它还提供了一个很好的示例,如何使用视图模型等(在我看来)-或者至少获得某种灵感。

摘自帖子(该属性将业务模型转换为视图模型):

[AutoMap(typeof(Product), typeof(ShowProduct))]
public ActionResult Details(int id)
{
    var product = _productRepository.GetById(id);

    return View(product);
}
于 2012-07-13T12:08:21.820 回答
0

正如其他答案中提到的,AutoMapper 是一个很好的解决方案。

但是,您也可以编写扩展方法来将您的实体转换为 ViewModel,反之亦然。也许这对于实体和模型类较少的小型项目更好。

例子:

public static class Extension
{
    public static MyViewModel ToModel(this MyEntity x)
    {
        return new MyViewModel
        {
            Id = x.Id,
            Name = x.FirstName + " " + x.LastName,
            UserId = x.User?.Id
        };
    }
}

并使用:

// NOTE: "MyEntityTable" is of type "MyEntity"

List<MyViewModel> toReturn = _db.MyEntityTable
                  .Include(x => x.User) // Include other tables to join.
                  .Select(x => x.ToModel())
                  .ToList();
于 2019-09-26T01:20:03.837 回答