原文

你不希望在controller里面出现任何领域知识

开发者经常有这样的疑问“这个代码应该放在哪呢?”应该使用仓储还是query类?....

怎么去实现职责分离和单一职责呢?

MediatR Notifications能帮助我们。

例子

假设你有一个简单的网站,有一个页面提供一个简单的联系我们的表单。

当客户完成了这个表单,会发送一封邮件给网站管理者。

public IHttpActionResult Post([FromBody]ContactUsForm contactUs)
{
var mailMessage = new MailMessage(contactUs.EmailAddress, "you@yoursite.com");
mailMessage.Subject = "Contact from web site";
mailMessage.Body = contactUs.Message; var smtp = new SmtpClient();
smtp.Send(mailMessage); return Ok();
}

现在你老板过来要求你将这个信息往数据库里面也放一份,以方便做一些报表统计。

然后你修改controller,加入了下面的一下代码:

public IHttpActionResult Post([FromBody]ContactUsForm contactUs)
{
var mailMessage = new MailMessage(contactUs.EmailAddress, "you@yoursite.com");
mailMessage.Subject = "Contact from web site";
mailMessage.Body = contactUs.Message; var smtp = new SmtpClient();
smtp.Send(mailMessage); var userMessage = new UserMessage {
Received = System.DateTime.Now,
UserEmailAddress = contactUs.EmailAddress,
UserFullName = contactUs.FullName
};
db.UserMessages.Add(userMessage);
db.SaveChanges(); return Ok();
}

现在单一职责被破坏了,但是你决定先这么处理不管了。

你老板是一个从来不容易满足的人,过了几天他又过来了,他希望将这个表单里面的客户信息存储到CRM系统中。

现在你的controller变成这样了:

public class ContactController : ApiController
{
using BlogSite.Models;
using BlogSite.Models.Crm;
using System.Net.Mail;
using System.Web.Http; namespace BlogSite.Controllers.Api
{
public class ContactController : ApiController
{
private ApplicationDbContext db = new ApplicationDbContext(); public IHttpActionResult Post([FromBody]ContactUsForm contactUs)
{
var mailMessage = new MailMessage(contactUs.EmailAddress, "you@yoursite.com");
mailMessage.Subject = "Contact from web site";
mailMessage.Body = contactUs.Message; var smtp = new SmtpClient();
smtp.Send(mailMessage); var userMessage = new UserMessage
{
Received = System.DateTime.Now,
UserEmailAddress = contactUs.EmailAddress,
UserFullName = contactUs.FullName
};
db.UserMessages.Add(userMessage);
db.SaveChanges(); var crm = new CrmInterface();
crm.Connect();
crm.SaveContact(new CrmContact { EmailAddress = contactUs.EmailAddress, FullName = contactUs.FullName }); return Ok();
}
}
}
}

评估/回顾(Taking stock)

这种需求还在不断的增加。每一个新需求都需要修改已有的controller。

这存在一个潜在的风险。对存在的代码的每次修改都有可能破坏现有的功能。

而且这些代码是同步执行的。如果你的CRM系统或者邮件服务器挂了或者变慢了,你的客户在提交表单的页面要等待许久。

分离职责

现在,很明显controller承担了太多的工作。有一种解决方案是将不同的职责放到他们自己的类中去。

using BlogSite.Services.Crm;
using BlogSite.Services.Data;
using BlogSite.Services.Email;
using System.Web.Http; namespace BlogSite.Controllers.Api
{
public class ContactController : ApiController
{
private IEmailService _emailService;
private IUserMessageService _userMessageService;
private ICrm _crm; public ContactController(IEmailService emailService, IUserMessageService userMessageService, ICrm crm)
{
_emailService = emailService;
_userMessageService = userMessageService;
_crm = crm;
} public IHttpActionResult Post([FromBody]ContactUsForm contactUs)
{
_emailService.Send(contactUs.EmailAddress, "you@yoursite.net", "Contact from web site", contactUs.Message);
_userMessageService.RecordUserMessageReceived(contactUs.EmailAddress, contactUs.FullName);
_crm.SaveContact(new CrmContact { EmailAddress = contactUs.EmailAddress, FullName = contactUs.FullName });
return Ok();
}
}
}

好些了,职责分离了,controller看起来整洁些了。

然后你还是在同步的运行(在这个例子中异步执行更好),controller和应用服务产生了耦合。如果要加什么新特性,你仍然需要修改controller。

使用MediatR Notifications

MediatR能给我们什么帮助呢?

比起在action里面写一些业务逻辑,MediatR可以让我们发布一个通知。

通知可以是异步或者同步的。

public class ContactController : ApiController
{
private IMediator _mediator; public ContactController(IMediator mediator)
{
_mediator = mediator;
} public async Task<IHttpActionResult> Post([FromBody]ContactUsForm contactUs)
{
var messageReceivedFromUserNotification = new MessageReceivedFromUserNotification {
EmailAddress = contactUs.EmailAddress,
FullName = contactUs.FullName,
Message = contactUs.Message,
SubmittedAt = DateTime.Now
};
await _mediator.PublishAsync(messageReceivedFromUserNotification); return Ok();
}
}

notification是一个通知,代表了一个通知的详细信息。

public class MessageReceivedFromUserNotification : IAsyncNotification
{
public DateTime SubmittedAt { get; set; }
public string FullName { get; set; }
public string EmailAddress { get; set; }
public string Message { get; set; }
}

接下来你需要为不同的action写不同的handler。

public class SaveUserMessage : IAsyncNotificationHandler<MessageReceivedFromUserNotification>
{
private ApplicationDbContext db = new ApplicationDbContext(); public async Task Handle(MessageReceivedFromUserNotification notification)
{
var userMessage = new UserMessage
{
Received = notification.SubmittedAt,
UserEmailAddress = notification.EmailAddress,
UserFullName = notification.FullName
};
db.UserMessages.Add(userMessage);
await db.SaveChangesAsync();
}
}
public class NotifySalesUserMessageReceived : IAsyncNotificationHandler<MessageReceivedFromUserNotification>
{
public async Task Handle(MessageReceivedFromUserNotification notification)
{
var mailMessage = new MailMessage(notification.EmailAddress, "you@yoursite.com");
mailMessage.Subject = "Contact from web site";
mailMessage.Body = notification.Message; var smtp = new SmtpClient();
await smtp.SendMailAsync(mailMessage);
}
}

[译]使用mediatR的notification来扩展的的应用的更多相关文章

  1. 「译」JUnit 5 系列:扩展模型(Extension Model)

    原文地址:http://blog.codefx.org/design/architecture/junit-5-extension-model/ 原文日期:11, Apr, 2016 译文首发:Lin ...

  2. [译]深入 NGINX: 为性能和扩展所做之设计

    来自:http://ifeve.com/inside-nginx-how-we-designed-for-performance-scale/ 这篇文章写了nginx的设计,写的很仔细全面, 同时里面 ...

  3. (译+注解)node.js的C++扩展入门

    声明:本文主要翻译自node.js addons官方文档.部分解释为作者自己添加. 编程环境: 1. 操作系统 Mac OS X 10.9.51. node.js v4.4.22. npm v3.9. ...

  4. 中介者模式及在NetCore中的使用MediatR来实现

    在现实生活中,常常会出现好多对象之间存在复杂的交互关系,这种交互关系常常是"网状结构",它要求每个对象都必须知道它需要交互的对象.例如,每个人必须记住他(她)所有朋友的电话:而且, ...

  5. C++: Virtual Table and Shared Memory

    See at: 补充栏3: C++对象和共享内存 (叙述内容和Link1的内容基本一致) <C++网络编程 卷1:运用ACE和模式消除复杂性> <C++ Network Progra ...

  6. [转]非常好的vsftpd安装于配置

    环境:CentOS 5.0 操作系统一.安装:1.安装Vsftpd服务相关部件:[root@KcentOS5 ~]# yum install vsftpd*Dependencies Resolved= ...

  7. 我见过最好的vsftpd配置教程(转)

    环境:CentOS 5.0 操作系统一.安装:1.安装Vsftpd服务相关部件:[root@KcentOS5 ~]# yum install vsftpd*Dependencies Resolved= ...

  8. vsftp、ftps 搭建

    今天公司某个产品预上线,该产品需要向政府某部门提供一些数据. 该部门提交数据需要使用ftps,苦逼的我只能是测试环境搭建一套,用来测试提交数据. 先自行科普下ftps. 一.搭建vsftp 安装vsf ...

  9. centos6.5下vsftpd服务的安装及配置并通过pam认证实现虚拟用户文件共享

    FTP的全称是File Transfer Protocol(文件传输协议),就是专门用来传输文件的协议.它工作在OSI模型的第七层,即是应用层,使用TCP传输而不是UDP.这样FTP客户端和服务器建立 ...

随机推荐

  1. 工作队列.py

    #对列模式图Work Queue背后的主要思想是避免立即执行资源密集型任务的时,需要等待其他任务完成.所以我们把任务安排的晚一些,我们封装一个任务到消息中并把它发送到队列,一个进程运行在后端发送并最终 ...

  2. CodeVS1288埃及分数(IDA*)

    在古埃及,人们使用单位分数的和(形如1/a的, a是自然数)表示一切有理数. 如:2/3=1/2+1/6,但不允许2/3=1/3+1/3,因为加数中有相同的. 对于一个分数a/b,表示方法有很多种,但 ...

  3. hdu 1238 Substrings(kmp+暴力枚举)

    Problem Description You are given a number of case-sensitive strings of alphabetic characters, find ...

  4. 2019 校内赛 RPG的天赋分支(贪心)

    Problem Description 很多游戏都有天赋树的概念,天赋树的不同分支具有不同的属性加成,那么合理选择分支就非常重要了.Luke最近沉迷一款RPG游戏,它的天赋树机制如下:角色具有n个可选 ...

  5. rdesktop ERROR: CredSSP: Initialize failed, do you have correct kerberos tgt initialized ? Failed to connect, CredSSP required by server

    错误信息: ERROR: CredSSP: Initialize failed, do you have correct kerberos tgt initialized ? Failed to co ...

  6. ACM-ICPC 2015 BeiJing

    比赛连接:ACM-ICPC 2015 BeiJing 本次比赛只写了 A G     然后 I题随后补 A 有一个正方形土地,上面有若干块绿洲.让你以x0为界限划一条竖线,要求左边绿洲面积>=右 ...

  7. Lisp经典算法

    求平方根 SUCCESSIVE AVERAGING DUE TO HERON OF ALEXANDRIA ** TO FIND AN APPROXIMATION TO SQRT(X) ** MAKR ...

  8. A1147. Heaps

    In computer science, a heap is a specialized tree-based data structure that satisfies the heap prope ...

  9. Java 读数据库字段时发现的一个现象

    早上发现有一个网名叫“帅!是不需要理由”的一个人,在后台只能看到“帅!是不需要理”,“由”字就是不显示出来. 经过分析发现,在Access数据库中,name这个字段的长度是15,因为我知道Access ...

  10. python学习笔记-列表和字典

    由于最近在看深度学习的代码,看到需要建立字典和列表来存储什么东西的时候,就想要去把字典和列表好好的了解清楚,其应用范围,差别,等等东西 首先我们来介绍,在python中存在如下的数据结构:列表list ...