原文

你不希望在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. Nginx优化文件编写

    server_tokens off; #并不会让nginx执行的速度更快,关闭它可隐藏错误页面中的nginx版本号charset utf-8,gbk; #字符#sendfile on;#tcp_nop ...

  2. [BJOI2012]最多的方案(记忆化搜索)

    第二关和很出名的斐波那契数列有关,地球上的OIer都知道:F1=1, F2=2, Fi = Fi-1 + Fi-2,每一项都可以称为斐波那契数.现在给一个正整数N,它可以写成一些斐波那契数的和的形式. ...

  3. Java String与Stringbuffer

    String 与其它类型的转换,e.g. BigInteger Stringbuffer 诸多函数,replace…… String 不能修改,Stringbuffer 可以修改, 应避免以下的操作: ...

  4. Day29--Python--缓冲区, 粘包

    tcp: 属于长连接,与一个客户端进行连接了以后,其他的客户端要等待.要想连接另外一个客户端,需要优雅地断开当前客户端的连接 允许地址重用:server.setsockopt(socket.SOL_S ...

  5. HTML学习笔记Day4

    一.浮动属性 1.首先要知道,div是块级元素,在页面中独占一行,自上而下排列,也就是传说中的流: 无论多么复杂的布局,其基本出发点均是:“如何在一行显示多个div元素”: 显然标准流已经无法满足需求 ...

  6. 百度富文本ueditor使用小结

    最近因工作需要使用了ueditor,根据自己的需求将开发使用时遇到的问题小结分享下. 1.可到官网根据自身情况下载最新版本,https://ueditor.baidu.com/website/ 2.h ...

  7. java十进制转三十六进制

    import java.util.HashMap; public class Ten2Thirty { private static final String X36 = "01234567 ...

  8. pt-kill 用法记录

    pt-kill 用法记录 # 参考资料Percona-Toolkit系列之pt-kill杀会话利器http://www.fordba.com/percona-toolkit-pt-kill.html ...

  9. 写给IT技术爱好者的一封信

    写给IT技术爱好者的一封信>当前运维素质的分析<... ---------------------- 虽相貌平平,但勤学苦练,亦收获颇丰!如果你决定要成为一名IT从业者,你需要承受以下的东 ...

  10. nGrinder TestRunner DNS / sun.net.spi.nameservice.NameServiceDescriptor

    s ngrinder3.3控制台验证脚本报错 http://ngrinder.642.n7.nabble.com/ngrinder3-3-td1301.html 目前发现3.3版本在控制台校验脚本报错 ...