翻译:Clean Architecture in .Net

前言

几年前我偶然发现了Robert Martin的一个关于分离关注点的演讲,在这个演讲的启发下,我尝试把在ASP.NET MVC应用中实践Robert提到的观点。

问题

PM:客户需要可以在线上订购产品。
我:好的,那么我需要把订单存在某个数据库里面,并且需要在页面上显示出来
PM:对了,如果库存没了,是不允许下单的。
我:你应该早点告诉我的... 我现在要添加新列用于过滤。

三个月后

PM:
为什么我们的bug那么多?
为什么我们的网站那么慢?
为什么这个改动要花这么多时间?

我承认这样的抱怨很公平,因为这些问题是我写的代码造成的。

项目总是从简单开始,然后(由于需求变动)逐渐变得复杂,最终变成一个“大泥球”(big ball of mud)。

我意识到,(直接把业务逻辑映射为具体实现的做法,)数据库通常会成为瓶颈,表现层逻辑混杂着业务逻辑和数据访问代码,这使得(单元/集成)测试变得非常困难。

解决办法

然后我看到了Uncle Bob的演讲Agility and Architecture (youtube链接),在演讲里面提到了另一种解决方法:"Clean Architecture"。在这种架构下,业务逻辑处于架构的中心位置,而web和数据库作为实现细节(位于架构外层)。

这个演讲解释了我之前碰到的问题,也给出了解决方案。但是,Uncle Bob并没有提供任何源码! 为了充分理解该架构,我需要一个具体的例子。我接下去看了很多Uncle Bob的书和视频课程,从中学到了很多。但还是在试图把这些原则应用到真实项目中时依然十分挣扎。

随后我尝试在实践中坚持其中的一些原则,发现效果确实不错。在这篇博客中我会阐述这些实践细节,并解释遵循的原则对实践的帮助,同时也会分享相应的代码样本。

Clean Architecture概述

在开始之前,我们先回顾一下Clean Architecture的关键元素。我不会对架构本身做详细地解释,如果你之前没有了解过,请先观看上面提到的视频,或者阅读Uncle Bob的博客The Clean Architecture译文)。

依赖规则

Clean Architecture的目的之一,是通过一种清晰的方式,把应用的业务逻辑封装起来。

上图展示了该架构,其中领域实体(Entity)和用例(Use case)处于“洋葱”的中心。

Uncle Bob:

用一组同心圆表示不同层级的软件代码,总的来说,越往内,抽象层次越高。最外层的圈代表的是机制级别的系统。最内层的代表的是策略级别的系统。

注意图中的依赖方向指出,代码依赖只能使由外向内。在实践中,这意味着你需要把描述业务逻辑的代码从具体交付机制(UI是web还是app,数据存放在数据库还是NoSql,等等)中分离出来。

这样能确保你的业务规则能独立存在,而不依赖于具体的框架选择和实现。

这样做有以下好处:

  • 分离关注点。
  • 业务规则代码只描述业务领域,不关注其他。
  • 在更换数据存储或web框架时,不需要改变业务逻辑代码。
  • 业务规则易于测试。
  • 业务规则易于重构。

向内依赖规则不仅限制业务规则,其他各处也都需要遵循。但隔离业务规则是Clean Architecture提供的最有价值的一点。

在实践中应该怎样做到呢?

练习

现在开始用一个具体例子来尝试实践这些原则。

业务需求

我们将从用例开始,这是很自然的做法。通常来说,开始一个项目是为了解决用户的具体问题。(在编写用例的时候,我们考虑)软件需要做什么事情,以及需要和用户如何交互。

Funda (作者的雇主)是荷兰的一个在线房地产平台。想象我们在开发一个新的功能,让某处特定房产的潜在客户联系对应的房产代理。

在Funda我们使用scrum。我们使用用户故事(User story)描述待实现功能的业务价值。

用户故事: 联系房产代理

作为顾客,我希望联系意向房产的代理,以了解该房产的更多细节。

然后我们会把User story展开为正式的用例。

用例: 联系房产代理

数据:

  • Email(必填,格式必须符合email规范)
  • PhoneNumber(必填)
  • HouseId(意向房产的Id)

主流程:

  • 客户发起 “联系房产代理”请求并提供上述数据。
  • 系统验证所有输入数据。
  • 系统把该请求记录下来,并在稍后通知房产代理。
  • 向用户展示处理结果(记录成功)。

异常流程: 验证失败

  • 终止处理该请求。
  • 向用户展示错误信息。

通常我们的开发人员并不需要把用例如此正规地形成文档。在本案例中我写成这样,是因为这有助于说明Clean Architecture的某些概念。

注意在用例中,我们只描述用户和系统的交互逻辑,而不会说明任何实现细节(如联系信息会被如何存储,UI如何呈现等)。这个用例可能会实现为控制台程序,电话程序或者网站,而用例并不负责确定实现方式。这正是问题的关键,用例用于描述“系统做什么”,但并不提及“系统怎么做”(Uncle Bob:业务逻辑不需要了解任何外部结构。)

交互器对象(Interactor)

在Clean Architecture中,每个用例的主流程通常可以映射为一个Uncle Bob称之为Interactor(交互器)的对象。Interactor以操作领域实体的方式描述用户和系统的交互模型。领域实体指的是应用里面的业务对象,例如本例中的Hourse。

根据以上创建的用例,我们写出了以下Interactor。依赖的ContactAgentRequestMessage,ContactAgentResponseMessage和ContactAgentRequestMessageValidator请参阅github源码,其中Request/Response Message均为POCO类,Validator用于验证参数正确性。

public class ContactAgentInteractor : 
    IRequestHandler<ContactAgentRequestMessage,ContactAgentResponseMessage>
{
    private readonly IRepository<int, House> _repository;
    private readonly IValidator<ContactAgentRequestMessage> _validator;

    public ContactAgentInteractor(
            IValidator<ContactAgentRequestMessage> validator, 
            IRepository<int, House> repository)
    {
        _validator = validator;
        _repository = repository;
    }

    public ContactAgentResponseMessage Handle(
            ContactAgentRequestMessage request)
    {
        var validationResult = _validator.Validate(request);
        if (validationResult.IsValid == false)
            return new ContactAgentResponseMessage(validationResult);

        var house = _repository.Get(request.HouseId);
        house.RegisterInterest(new Interest
        {
            CustomerEmailAddress = request.CustomerEmailAddress,
            CustomerPhoneNumber = request.CustomerPhoneNumber,
            CreationDate = DateTime.Now
        });

        _repository.Save(house);

        return new ContactAgentResponseMessage(
                        validationResult, 
                        request.HouseId);
    }
}

这个Interactor里面有几个点需要注意:

  1. 几乎是一对一地翻译了之前定义的用例。
  2. 使用了简单消息(plain request/response message,plain指的是POCO对象,区别于Http Request/Response这种为某种context定义的对象)作为输入/输出,并没有任何ASP.NET的引用。
  3. 依赖于IRepository。

上述Interactor只是按功能规格,简明地描述了系统应该如何运作(以支持该功能需求)。

Interactor通过plain message接收参数和返回结果,这样做可以把Interactor从使用它的应用中解耦出来。Interactor只需要知道Request/Response Message,不再需要知道其他系统实现细节。

Interactor通过Repository(在洋葱图中称之为Gateway)获得实体,然后对其进行操作,再通过Repository保存修改后的实体。这个依赖顺序是违反了向内依赖规则的,但我们可以引入接口,并通过使用依赖倒置原则来处理这种情况。

如上图所示,Interactor依赖定义在实体层的IRepository接口。该接口定义了实体的读/写功能,但不提供具体实现。Interactor对象并不依赖该接口在接口适配层(Interface Adapters)的具体实现。

领域实体(Entity)

上面我们提到了领域实体,比如本例中的Hourse,现在我们对此做进一步讨论。

Uncle Bob:

领域实体用于封装公司的业务规则。 一个实体可以是有方法的对象,或者是一组数据结构和函数,这并不重要,只要确保实体可以被用在公司各个不同的应用中。

如果你只是写一个独立的应用(而不需要考虑在企业内共享),那么领域实体就是该应用的业务对象。领域实体封装了上层业务规则,通常来说,在外部环境变动的时候,这些实体对象是不应被改变的。

例如,你不会期望实体会被页面导航,或者安全方面的改动所影响。在任何应用中,操作层的变更都不应该影响领域实体层。

如何为业务领域建模是另一个大的话题,我在这里不做太多讨论。不过(在建模完成后),实现是很直接了当的:在本例中,我使用下面的POCO类描述应用的业务对象和业务规则。

public class House
{
    public int? Id { get; set; }
    public string Address { get; set; }
    public IList<Interest> Leads { get;  }

    public House()
    {
        Leads = new List<Interest>();
    }

    public void RegisterInterest(Interest interest)
    {
        Leads.Add(interest);
    }
}
public class Interest
{
    public string CustomerEmailAddress { get; set; }
    public long CustomerPhoneNumber { get; set; }
    public DateTime CreationDate { get; set; }
}

在Funda,我们公布房源信息,而客户可以关注对特定房源,这就是我们的业务模型。而使用领域模型(而不是具体的web应用实现)来描述该业务模型,这样的做法非常简洁。保持这部分代码简单明了,在重构系统或添加新功能的时候,能提供更高的灵活度。

我承认这是一个非常简单甚至简陋的领域模型的例子(有人称之为贫血模型),但我希望通过这个例子可以说明:

你确实可以通过POCO类来描述业务应用的运行行为,然后你可以在应用中直接引用它们,从而避免把业务逻辑混淆到处理web,数据库,消息队列或其他“交付机制”(delivery mechanisms)的代码中。

任务完成!

嗯,真完成了,某种意义上……

回顾一下,我们已经实现了:

  • Interactor对象
  • Request/Response消息
  • 业务领域Entity

依赖它们,你已经可以对项目待解决的问题进行建模,描述以及单元测试。虽然它还不是一个能工作的web应用。我并不是在开玩笑,在某种意义上,业务逻辑确实是可以在实现任何交付机制(web,database,etc.)之前完成的。

对我来说,一点关键的认知是,在实现了这些基础后,你可以为你的项目或者准备要解决的问题开始建模了。你可以为这些类写单元测试或者console应用,以证明这些业务逻辑是满足需求的。事实上,我在源码里面确实加入了一个用于测试的console应用。

下面我会简略地谈及如何把这些已有部分集成到web应用中。但我希望现在已经说服你相信,web应用可以搭建在我已经演示的Clean Architecture之上。

集成ASP.NET MVC应用

如果你希望通过ASP.NET MVC应用来呈现我们已经实现的业务逻辑,接下去要写的代码将会落在Clean Architecture的接口适配层(Interface Adapter)。

Uncle Bob:

这一层的软件结构的目的就是进行数据的转换,将便于用户实例和实体层操作的数据结构变化成为最便于外部结构(比如数据库或者Web)操作的数据结构。比如GUI的MVC结构,表现器、视图器、控制器都是属于这个结构的。这层很可能是通过控制器将数据结构传给用户实例层,并且返回数据给表现器,视图器。

实际上大部分的web应用具体实现能在这层找到。

当我们的客户想通过网站联系房产代理,通常他们需要填写一些HTML表单。然后模型绑定器(model binder)会把表单数据转换为强类型的ContactAgentRequestMessage对象,并交由AgentController处理该请求:

public class AgentController
{
    private readonly ContactAgentInteractor _interactor;
    private readonly ContactAgentResponsePresenter _presenter;

    public AgentController(ContactAgentInteractor interactor, 
                            ContactAgentResponsePresenter presenter)
    {
        _interactor = interactor;
        _presenter = presenter;
    }

    public void Contact(ContactAgentRequestMessage requestMessage)
    {
        var response = _interactor.Handle(requestMessage);
        var viewModel = _presenter.Handle(response);
        var view = new ConsoleView(viewModel);
        view.Render();
    }
}

现在我们必须越过边界进入内层,也就是业务逻辑的所在。我们通过调用Interactor对象的Handle方法做到了这点,在这里,我们需要先考虑清楚,在方法调用中,我们应该使用什么样的数据结构传递参数。

Uncle Bob:

一般跨层调用的数据是简单的数据结构。你可以使用数据结构或者是简单的数据传输流,又或者可以通过函数的参数来进行传递。你也可以将数据封装到一个hashmap结构,或者一个对象中。数据传输最重要的事情是无依赖,简单。我们并不希望跨层传递的数据是实体,或者是数据表的行数据,理由是我们不希望数据有任何形式的违反依赖规则。

...

当我们跨层传递数据的时候,我们应该以便于内圈使用的数据格式传递。

最后一句是最关键的,重复一下:

“当我们跨层传递数据的时候,我们应该以便于内圈使用的数据格式传递。”

在ASP.NET的世界里,如同Visual Studio的示例,人们倾向(基于设计好的数据库)使用Entity Framework创建实体类,并且把它们用作view model,domain entity,DTO等等。人们也会很自然地使用它们在Controller和Interactor之间传递数据,但这违反了向内依赖规则。

与之相反,内圈应该定义输入参数的数据格式。在我们的例子中,作为Interactor和Controller交互的参数,ContactAgentRequestMessage类同样被定义在了业务逻辑层中。这样做就遵循了向内依赖规则。

最后,Interactor对象完成它自己的工作。执行一系列的验证逻辑,然后把数据保存下来(保存动作通过IRepository接口执行,而具体的保存工作由外圈的IRepository实现来完成)。

Uncle Bob:

在接口适配层,数据在这层会被转换,将便于实体层和用户实例层使用的数据转化成为持久层能使用的数据,比如数据库。当一些外部的服务需要与用户实例层和实体层进行交互的时候,这时候需要的数据转换也理所当然地放在这一层了。

在本例中我简单地实现了一个假的Repository,测试的时候你不需要安装和配置任何数据库。

public class InMemoryHouseRepository : IRepository<int, House>
{
    private static readonly Dictionary<int, House> Store = 
    new Dictionary<int, House>();
    
    House IRepository<int, House>.Get(int id)
    {
        return Store[id];
    }

    public House Save(House entity)
    {
        if (entity.Id.HasValue == false)
            entity.Id = Store.Count;

        Store[entity.Id.Value] = entity;

        return entity;
    }
}

在实际项目中,通常会使用ORM框架来实现Repository。我喜欢使用NHibernate,有时候也会考虑使用Dapper。

表现层

在业务逻辑处理完成后,Interactor对象将返回一个ContactAgentResponseMessage消息给Controller,use case的任务到此结束。消息对象只包含最终用户必须的信息,它实际上是一种数据结构,符合use case描述的功能需求,适合被Interactor调用。

public class ContactAgentResponseMessage
{
    public ValidationResult ValidationResult { get; }
    public long? HouseId { get; private set; } 
    
    public ContactAgentResponseMessage(
        ValidationResult validationResult, 
        int? houseId =null)
    {
        HouseId = houseId;
        ValidationResult = validationResult;
    }
}

然后我们希望以友好的界面展示给用户,我们会使用Presenter对象把原始的ResponseMessage换行为ViewModel。ViewModel也是一种简单数据结构,但它的结构是为适配View而设计的。

例如,我们希望展示友好的操作成功或失败的消息:

public class ContactAgentResponsePresenter
{
    public ContactAgentResponseViewModel Handle(ContactAgentResponseMessage responseMessage)
    {
        switch(responseMessage.ValidationResult.IsValid)
        {
            case true:
                return new ContactAgentResponseViewModel(Texts.ThankYou);
            case false:
                var sb = new StringBuilder();
                sb.AppendLine(Texts.ValidationError);

                foreach (var error in responseMessage.ValidationResult.Errors)
                {
                    sb.AppendLine(error.ErrorMessage);
                }
                return new ContactAgentResponseViewModel(sb.ToString());
        }
        return null;
    }
}
public class ContactAgentResponseViewModel
{
    public string Text { get; private set; }

    public ContactAgentResponseViewModel(string text)
    {
        Text = text;
    }
}

最后,Controller把ViewMode传递给视图引擎,引擎将用于渲染视图并展示给用户。

任务完成!

结论

遵循“向内以来规则”可以避免业务逻辑被外包框架和实现的变更所影响。这样做可以保持业务逻辑独立整洁,易于测试和修改。

在这个(业务实现的)基础上,你可以根据用户的需要,选择以web或者其他类型的应用方式来交付系统。

我提供了一个console应用作为例子,展示在.Net中如何实践这个理论。你可以在我的github上找到该例子,其中包含本文展示的所有源码。

感谢Uncle Bob,Rodi Evers和Michiel van Oosterhout提供灵感!

源文链接Clean Architecture in .Net (翻译:woodylic

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,602评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,442评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,878评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,306评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,330评论 5 373
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,071评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,382评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,006评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,512评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,965评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,094评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,732评论 4 323
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,283评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,286评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,512评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,536评论 2 354
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,828评论 2 345

推荐阅读更多精彩内容