原文

你不希望在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. iptables(2)

    MASQUERADE同样是做源地址转换,只不过防火墙会根据该策略自动查找可用的公网IP地址,适应变化的情况•若接口使用ppp+,表示匹配ppp0.ppp1……中任意可用的拨号连接•若需要演示操作,可以 ...

  2. Arch Linux中安装Anaconda

    安装步骤 通过AUR安装yaourt -S anaconda 激活Anaconda环境source /opt/anaconda/bin/activate root 关闭Anaconda环境source ...

  3. 解题:NOI2018 你的名字(68pts暴力)

    题面 rt,如果省选没退役就补 SAM的优势:简单明了 先建S的SAM并标记所有节点,之后每次询问直接把T按广义SAM的方法插上去,统计新加的节点到根的状态代表的本质不同子串数,减掉被标记的部分就是T ...

  4. 第二篇 - python爬取免费代理

    代理的作用参考https://wenda.so.com/q/1361531401066511?src=140 免费代理很多,但也有很多不可用,所以我们可以用程序对其进行筛选.以能否访问百度为例. 1. ...

  5. Java 存储时间戳的几种方式

    有时需要记录一下数据生成时间的时间戳,精确到秒,这里记录一下java存储时间戳字符串的几种方式 1.DateFormat private static final SimpleDateFormat s ...

  6. 第二十七节,IOU和非极大值抑制

    你如何判断对象检测算法运作良好呢?在这一节中,你将了解到并交比函数,可以用来评价对象检测算法. 一 并交比(Intersection over union ) 在对象检测任务中,你希望能够同时定位对象 ...

  7. POJ 3463 Sightseeing (次短路经数)

    Sightseeing Time Limit: 2000MS   Memory Limit: 65536K Total Submissions:10005   Accepted: 3523 Descr ...

  8. maven pom添加本地jar,不提交私库

    <dependency> <groupId>taobao-sdk</groupId> <artifactId>taobaosdk</artifac ...

  9. Go数据类型和变量

    一:Go数据类型 1.1 Go语言按照分类有以下几种数据类型 布尔型 布尔型的是一个常量true或者false 数字类型 整型int和浮点型 float32, float64 字符串类型 字符串就是一 ...

  10. Java基础super关键字、final关键字、static关键字、匿名对象整理

    super关键字 10.1子父类中构造方法的调用 public class Test { public static void main(String[] args) { new Zi(); } } ...