问题 MVC模式差异


我只需要一些链接到我可以阅读的文章或关于MVC(C#)中使用的不同模式的一些基本解释。

目前,我倾向于使用视图模型模式构建我的Web应用程序。对于每个视图,我都有一个视图模型。我喜欢这种方法纯粹是因为模型中不需要那么多垃圾,我可以在这里使用一些基本的数据注释。

我现在也在视图模型中构建我的视图模型(不确定这是否正确?)这样我就可以让控制器尽可能简单。

有时候我发现自己在我的控制器中添加了很多逻辑,我认为这对我来说也很好,这就是控制器的用途。

现在根据上述内容,我说我可以很愉快地构建我的应用程序,没有任何重大问题。然而,在我正常浏览代码示例等的同时,我经常发现有很多其他方法可供不同的开发人员用来完成我上面所做的基本操作,并且我想要解释它们都适合在一起。

我经常看到提到“使用你的存储库做等等”。我确实使用了“有时”的存储库,但这主要是为了模型查询,我知道我将来会重复使用它总是会变成一点点垃圾场。这里的最佳做法是什么?

我也看到提到“接口”和“服务层”我在这里完全迷失了......大多数例子对我来说似乎只是添加了越来越多的步骤来实现相同的目标。它们是如何使用的?


2848
2017-12-15 08:00


起源



答案:


我不能说这是最好的做法,但这就是我使用的,为什么,我们在这里:


1.存储库。

它们的结构如下:

有三个基本接口, IRead<>IReadCreate<> 和 IReadCreateDelete<>

interface IRead<T>
{ 
    T FindOne(int id);
    IQueryable<T> GetOne(int id);
    IQueryable<T> FindAll(Expression<Func<T, bool>> predicate);
}

interface IReadCreate<T> : IRead<T>
{ 
    T Create();
    void Create(T entity);
}

interface IReadCreateDelete<T> : IReadCreate<T>
{ 
    void Delete(int id);
    void Delete(T entity);
    void DeleteWhere(Expression<Func<T, bool>> predicate);
}

所有其他接口如下所示:

interface ICategoriesRepository : IReadCreate<Category>
{
    IQueryable<Category> GetAllActive();
}

所有这些都为它们所依赖的数据源提供了额外的有用功能。这意味着,我无法访问我的实现存储库中的其他类型存储库。应该这样做 服务。 (往下看。)

这种方法的主要目标是显示调用代码(来自另一个程序集,因为我的所有存储库,服务和其他契约都在单独的DLL项目中定义(作为接口))它能做什么(比如读取和创建项目)以及什么它不能做(比如删除项目)。


2.服务

服务和实施业务逻辑的最佳方式。他们应该实施所有重要的逻辑方法。为了实现这种实现,他们需要一些存储库依赖,并且在这里它来了 Dependency Injector。我更喜欢使用 Ninject,因为它允许我注入这样的依赖属性:

internal class CategoriesService : ICategoryService
{
    public ICategoriesRepository CategoriesRepository { get; set; }
    public IWorkstationsRepository WorkstationsRepository { get; set; }

    // No constructor injection. I am too lazy for that, so the above properties 
    // are auto-injected with my custom ninject injection heuristic.

    public void ActivateCategory(int categoryId)
    {
        CategoriesRepository.FindOne(categoryId).IsActive = true;
    }
}

服务的目标是从控制器和存储库中消除业务逻辑。


3. ViewModels

很酷的事情,正如你所说的那样,但原因在于你为什么在他们自己建造它们是我无法得到的东西。我正在使用 automapper 对于它(具有可查询的扩展名),它允许我创建这样的视图:

假设我有一个 视图 需要一个 IEnumerable<TicketViewModel> 模型。我所做的是:

public class FooController : Controller
{
     public IMappingEngine Mapping { get; set; } // Thing from automapper.
     public ITicketsRepository TicketsRepository { get; set; }

     public ViewResult Tickes()
     { 
         return View(TicketsRepository.GetAllForToday().Project(Mapping)
             .To<TicketViewModel>().ToArray();
     }
}

而已。对存储库的简单调用,它调用底层数据源(另一种模式。我不会写它,因为它的抽象仅用于测试。),这使得调用数据库(或任何你实现的 IDataSource<T>)。 Automapper自动映射 Ticket 至 TicketViewModel 和表单数据库我检索 我的ViewModel列唯一需要的,包括单个请求中的交叉表


结论

还有很多话要说,但我希望这会给你一些思考的东西。我使用的所有模式和程序是:

  1. 自动映射(映射);
  2. Ninject(依赖注入);
  3. 存储库(数据访问);
  4. 数据源(数据来自......井......来自数据源);
  5. 服务(数据交互);
  6. ViewModels(数据传输对象);
  7. 也许其他我会编辑添加的东西。

5
2017-12-15 08:37



+1实际代码! - Michael Viktor Starberg
谢谢你的信息。根据这个和其他评论,我肯定错过了服务层。我目前正在使用linq2sql,因为我是mvc和c#的新手,所以我也不想也用EF煎炸我的大脑。您的任何评论都会根据linq2sql的用法而改变吗?我假设不是因为它只是数据访问的一部分,但我想我会检查。 - munkee
@munkee我的评论实际上是基于l2s。 - AgentFire
谢谢,最后我还没有使用automapper知道,但它是否处理需要来自多个表的列和创建选择列表等的视图模型?如果是这样的话,请考虑一下它,因为代码中的所有内容看起来都更清晰 - munkee
@munkee是的。正如我在帖子中提到的那样。 - AgentFire


答案:


我不能说这是最好的做法,但这就是我使用的,为什么,我们在这里:


1.存储库。

它们的结构如下:

有三个基本接口, IRead<>IReadCreate<> 和 IReadCreateDelete<>

interface IRead<T>
{ 
    T FindOne(int id);
    IQueryable<T> GetOne(int id);
    IQueryable<T> FindAll(Expression<Func<T, bool>> predicate);
}

interface IReadCreate<T> : IRead<T>
{ 
    T Create();
    void Create(T entity);
}

interface IReadCreateDelete<T> : IReadCreate<T>
{ 
    void Delete(int id);
    void Delete(T entity);
    void DeleteWhere(Expression<Func<T, bool>> predicate);
}

所有其他接口如下所示:

interface ICategoriesRepository : IReadCreate<Category>
{
    IQueryable<Category> GetAllActive();
}

所有这些都为它们所依赖的数据源提供了额外的有用功能。这意味着,我无法访问我的实现存储库中的其他类型存储库。应该这样做 服务。 (往下看。)

这种方法的主要目标是显示调用代码(来自另一个程序集,因为我的所有存储库,服务和其他契约都在单独的DLL项目中定义(作为接口))它能做什么(比如读取和创建项目)以及什么它不能做(比如删除项目)。


2.服务

服务和实施业务逻辑的最佳方式。他们应该实施所有重要的逻辑方法。为了实现这种实现,他们需要一些存储库依赖,并且在这里它来了 Dependency Injector。我更喜欢使用 Ninject,因为它允许我注入这样的依赖属性:

internal class CategoriesService : ICategoryService
{
    public ICategoriesRepository CategoriesRepository { get; set; }
    public IWorkstationsRepository WorkstationsRepository { get; set; }

    // No constructor injection. I am too lazy for that, so the above properties 
    // are auto-injected with my custom ninject injection heuristic.

    public void ActivateCategory(int categoryId)
    {
        CategoriesRepository.FindOne(categoryId).IsActive = true;
    }
}

服务的目标是从控制器和存储库中消除业务逻辑。


3. ViewModels

很酷的事情,正如你所说的那样,但原因在于你为什么在他们自己建造它们是我无法得到的东西。我正在使用 automapper 对于它(具有可查询的扩展名),它允许我创建这样的视图:

假设我有一个 视图 需要一个 IEnumerable<TicketViewModel> 模型。我所做的是:

public class FooController : Controller
{
     public IMappingEngine Mapping { get; set; } // Thing from automapper.
     public ITicketsRepository TicketsRepository { get; set; }

     public ViewResult Tickes()
     { 
         return View(TicketsRepository.GetAllForToday().Project(Mapping)
             .To<TicketViewModel>().ToArray();
     }
}

而已。对存储库的简单调用,它调用底层数据源(另一种模式。我不会写它,因为它的抽象仅用于测试。),这使得调用数据库(或任何你实现的 IDataSource<T>)。 Automapper自动映射 Ticket 至 TicketViewModel 和表单数据库我检索 我的ViewModel列唯一需要的,包括单个请求中的交叉表


结论

还有很多话要说,但我希望这会给你一些思考的东西。我使用的所有模式和程序是:

  1. 自动映射(映射);
  2. Ninject(依赖注入);
  3. 存储库(数据访问);
  4. 数据源(数据来自......井......来自数据源);
  5. 服务(数据交互);
  6. ViewModels(数据传输对象);
  7. 也许其他我会编辑添加的东西。

5
2017-12-15 08:37



+1实际代码! - Michael Viktor Starberg
谢谢你的信息。根据这个和其他评论,我肯定错过了服务层。我目前正在使用linq2sql,因为我是mvc和c#的新手,所以我也不想也用EF煎炸我的大脑。您的任何评论都会根据linq2sql的用法而改变吗?我假设不是因为它只是数据访问的一部分,但我想我会检查。 - munkee
@munkee我的评论实际上是基于l2s。 - AgentFire
谢谢,最后我还没有使用automapper知道,但它是否处理需要来自多个表的列和创建选择列表等的视图模型?如果是这样的话,请考虑一下它,因为代码中的所有内容看起来都更清晰 - munkee
@munkee是的。正如我在帖子中提到的那样。 - AgentFire


当我开始阅读你的帖子时,我在想,也许你正在寻找的是对SOLID原则的理解。最后提到接口和服务层。有趣。

有很多文章庆祝SOLID和DRY的圣杯(很多人都没有理解DRY倡导者真正提出的建议)。但是.NET世界中的一般想法,不是去aspx中自动生成的Page_Load,而是开始输入所有内容,直到页面执行它应该做的事情。 MVC来救援。

你说每个视图都有一个模型。我会说那个声音。即使两个模型相同,它们也只是相同,不一样。例如:NewsItem不是EventItem。如果你想扩展一个,它不应该影响另一个。

然后,您继续说您正在视图模型中生成模型。这听起来倒退了。但是你说你是这样做的,以保持你的控制器清洁。好!您的思维方式缺少的是服务。

您要做的是将实际执行任何类型工作的所有代码移动到服务中。服务可以基于方面,或基于特征,或者为什么不是控件。现在看一个Web项目,我看到:VisitorService,NewsfeedService,CalendarService,CachingService,MainMenuService,HeaderService,FooterService等等无限制地。

在这种情况下,控制器仅负责向模型询问执行某些工作的服务(或服务)。然后将该模型转发到视图。

一旦您将“业务逻辑”转换为服务,您就可以轻松地将IoC(控制反转)应用于您的项目,如果这让您满意的话。我还没有对IoC投票。我有可怕的好处并不像广告那么好,你可以确保没有代码臃肿。但IoC确实要求你在编码前思考。

关于IoC的一个非常简单的教程,我推荐Ninject。它不仅有Ninjas,还有武士,剑和手里剑。这比汽车和动物要凉爽得多。

https://github.com/ninject/ninject/wiki/Dependency-Injection-By-Hand


3
2017-12-15 08:52



感谢您准确了解我一直在寻找的信息。我将通过服务清理我的一些工作。我认为IOC不是必不可少的,我可以在以后回来吗? - munkee
确实。您可能想要做的是首先将所有逻辑都纳入服务。下一步可能是将您的Linq2Sql层变成一个只有服务可以与之交谈的类库。之后,您可以将依赖于数据层的服务转换为接口,并尝试使用IoC的ninject。最后,您可以进行单元测试来模拟数据库。 - Michael Viktor Starberg
感谢关于如何开始重新处理我的一些代码的指示。如果我要开始一个新项目,那么所有这些层中有多少值得一试。目前,我们的业务只是冒险进入网络应用程序,特别是使用mvc c#。在可预见的未来,我们坚持使用MS SQL Server,我只能假设人们将应用程序底层数据库更改为完全不同的设置是非常罕见的情况。如果这是现实世界的案例,这些层中的一些是矫枉过正或仅仅是良好的实践(用mock进行单元测试真的是一大优点)? - munkee
将所有业务逻辑转化为服务可以极大地提高代码质量。即使在我的WebForms项目中,我也遵循这种模式。至于IoC;我是代码合约的忠实粉丝,但他们是PITA添加到接口,为什么IoC和CC不能很好地混合。但是,IoC的想法不是为您的数据库提供Oracle支持。而是能够将数据库的某些方面切换为可预测值的单元测试的简单类。 - Michael Viktor Starberg
我刚刚开始阅读adam freeman和steven sanderson的pro asp.net mvc 3框架,它确实将你所说的一切都放到了上下文中。感谢所有的帮助,现在有时间做一些学习! - munkee


控制器:

理论上,您的控制器应该只处理“数据”。将信息从一个地方移动到另一个地方。

小例子:

  1. 控制器接收带有一些参数的请求“GetMessage”。
  2. 将此数据发送到服务层。在您正在访问的服务层中 存储库返回消息。
  3. Cntroller收到此消息(或者如果没有null)并决定是否重新发送收到的消息或者可能存在错误,并且应该以某种方式通知用户。

所有“理论上”的业务逻辑都应该落后于某个服务层。这样你就可以轻松测试一切。控制器中的逻辑使得一些测试更加困难。

接口:

基于接口的设计现在非常流行。特别是所有IOC Containers处理依赖注入。但是,如果您从这个概念开始,请不要理会这些关键字。如果您知道Repository模式,那么首先尝试使用IRepository接口,而不是通过具体类访问存储库,使用IRepository。 (只需将控制器中的字段从Repository更改为IRepository)。

一般关于接口

您将在更复杂的场景中看到界面的好处,但有一种技术可以向您展示这种方法的所有荣耀。单元测试+模拟。


2
2017-12-15 08:43



我不得不说我开始问自己发布的问题的原因之一是因为我想让自己开始学习单元测试(我只用了几个月的c#和mvc)而且我明白了当前的结构是一个完整的混乱尝试和测试。 - munkee
这是好方法:)你知道你能做什么吗?阅读如何在NUnit等框架中编写第一个单元测试,并尝试为其中一个控制器编写一些测试:)。通过单元测试,您可以通过分析创建测试场景的容易程度来轻松检查代码质量。如果您需要1000行代码来初始化某些特定场景,您会立即注意到某些问题。你知道“测试”只是TDD的副产品,编写测试的主要重点是检查你的设计和代码结构。通过使用您的代码,您可以获得有关质量的快速反馈。 - Michal Franc
我已经离开并正确运行接口和存储库。我现在正在尝试实现我的服务层。我是否正确地认为我的控制器既可以引用存储库的接口,也可以引用服务层。例如,由于仅通过我的界面从存储库中检索数据,因此有时可能不需要服务层。当我需要我的服务层时,它将引用接口以确保检索/操作数据,然后基本上将信号传回给我的控制器,一切正常/不能让登录继续进行? - munkee
嗯,通常存储库位于服务层内。在发送到视图之前,应使用服务层来处理从DB接收的原始数据。例如,您可以在此处将模型转换为ViewModel。你的方法还可以。每种方法都可以,这取决于具体情况。 Tbh你无法创造完美的代码。获得一些工作状态,即使你失败了,你也会知道在这个特殊情况下不需要这个设计,或者你缺少某些东西。 - Michal Franc
好的,如果我有这个权利,我会:视图 - >控制器 - >服务层,可能/可能不会使用存储库 - >数据库。如果我希望我可以在控制器和服务之间以及服务和存储库之间建立接口?然后,如果我愿意,我可以将这些接口用于moq,但我也可以对上述“层”中的每个人进行单元测试。目前,我的存储库位于我的域层中。所以我有例如App.WebUI,App.Services,App.Domain。 - munkee