原文

CQRS

我是CQRS模式的粉丝。对我来说CQRS能让我有更优雅的实现。它同样也有一些缺点:通常需要更多的类,workflow不总是清晰的。

MediatR

MediatR的文档非常不错,在这就不重复了。但是为了有个基本的了解,在这还是举个小例子,来看看command是怎么被处理的:

[Test]
public void ItShouldHandleBasicCommands()
{
var mediator = GetMediator(); var command = new Command();
var response = mediator.Send(command); response.Should().NotBeNull();
}

MediatR接收一个command,让后将它发送到适当的handler去:

public class Command : IRequest<Response>
{
} public class CommandHandler : IRequestHandler<Command, Response>
{
public Response Handle(Command message)
{
return new Response();
}
}

注册command和command handler非常简单。下面的例子是关于使用Ninject基于约定来注册。找到所有的handler接口,绑定它们:

kernel.Bind(scan => scan.FromThisAssembly()
.SelectAllClasses()
.Where(o => o.IsAssignableFrom(typeof(IRequestHandler<,>)))
.BindAllInterfaces());

下面是一些其他的注册,它们也非常重要。注册IMediator。factory实例的调用告诉MediatR如何解析单个和多个实例。然后注册MediatR。

kernel.Bind<SingleInstanceFactory>()
.ToMethod(context => (type => context.Kernel.Get(type)));
kernel.Bind<MultiInstanceFactory>()
.ToMethod(context => (type => context.Kernel.GetAll(type)));
var mediator = kernel.Get<IMediator>();

Validation and Decorators

装饰模式可以让我们在不修改对象的前提下,增加这个对象的行为。一个通常的做法是添加输入验证。使用装饰器包装一个command handler,以使得可以在使用前验证这个command。下面的command和command handler简单的返回一个response。

public class Foo : IRequest<Response>
{
public string Message { get; set; }
} public class FooHandler : IRequestHandler<Foo, Response>
{
public Response Handle(Foo message)
{
return new Response();
}
}

FluentValidation是个非常不错的验证包。首先我们需要一个validation类,然后我们需要用Ninject注册它。

下面是一个简单的验证器。判断Message属性是否为空。如果为空,返回一个错误:

public class FooValidator : AbstractValidator<Foo>
{
public FooValidator()
{
RuleFor(ping => ping.Message).NotEmpty();
}
}

使用Ninject注册验证器非常简单。下面的代码绑定了程序集中所有的验证器:

kernel.Bind(scan => scan.FromThisAssembly()
.SelectAllClasses()
.InheritedFrom(typeof(AbstractValidator<>))
.BindAllInterfaces());

下一步是使用这个validator。下面的代码来自Jimmy Bogard的网站。是一个实现用来在命令发送到handler前验证命令的装饰类的例子:

public class ValidatingHandler<TRequest, TResponse> : IRequestHandler<TRequest, TResponse>
where TRequest : IRequest<TResponse>
{
private readonly IRequestHandler<TRequest, TResponse> handler;
private readonly IValidator<TRequest> validator; public ValidatingHandler(IRequestHandler<TRequest, TResponse> handler, IValidator<TRequest> validator)
{
this.handler = handler;
this.validator = validator;
} [DebuggerStepThrough]
public TResponse Handle(TRequest message)
{
var validationResult = validator.Validate(message); if (validationResult.IsValid)
return handler.Handle(message); throw new ValidationException(validationResult.Errors);
}
}

下一步是如何配置Ninjec来创建一个handler和装饰它。Binding Decorators - Mediators with Ninject这篇文章讲的非常好。"注册handler。当验证handler创建后,注入一个handler。当handler被请求返回一个validating handler。"

当Ninject被要求创建一个handler,它首先创建一个calidating handler。

kernel.Bind(scan => scan.FromThisAssembly()
.SelectAllClasses()
.Where(o => o.IsAssignableFrom(typeof(IRequestHandler<,>)))
.BindAllInterfaces()); kernel.Bind(scan => scan.FromThisAssembly()
.SelectAllClasses()
.InheritedFrom(typeof(IRequestHandler<,>))
.BindAllInterfaces()
.Configure(o => o.WhenInjectedInto(typeof(ValidatingHandler<,>)))); kernel.Bind(typeof(IRequestHandler<,>)).To(typeof(ValidatingHandler<,>));

下面看看如果command的message是空和不是空的时候是什么样子的。

[Test]
public void ItShouldProcessCommands()
{
var mediator = GetMediator(); var command = new Foo { Message = "valid ping" };
var response = mediator.Send(command); response.Should().NotBeNull();
} [Test]
public void ItShouldValidateTheCommand()
{
var mediator = GetMediator(); var ping = new Foo();
Action act = () => mediator.Send(ping); act.ShouldThrow<ValidationException>();
}

本文的源码位于github

[译]MediatR, FluentValidation, and Ninject using Decorators的更多相关文章

  1. Fluent Validation + NInject3 + MVC5

    Fluent Validation + NInject + MVC - Why & How : Part 1 http://fluentvalidation.codeplex.com/ htt ...

  2. 使用.NET 6开发TodoList应用(11)——使用FluentValidation和MediatR实现接口请求验证

    系列导航及源代码 使用.NET 6开发TodoList应用文章索引 需求 在响应请求处理的过程中,我们经常需要对请求参数的合法性进行校验,如果参数不合法,将不继续进行业务逻辑的处理.我们当然可以将每个 ...

  3. [译]ASP.NET Core中使用MediatR实现命令和中介者模式

    作者:依乐祝 原文地址:https://www.cnblogs.com/yilezhu/p/9866068.html 在本文中,我将解释命令模式,以及如何利用基于命令模式的第三方库来实现它们,以及如何 ...

  4. [译]使用mediatR的notification来扩展的的应用

    原文 你不希望在controller里面出现任何领域知识 开发者经常有这样的疑问"这个代码应该放在哪呢?"应该使用仓储还是query类?.... 怎么去实现职责分离和单一职责呢? ...

  5. [译]使用Command模式和MediatR简化你的控制器

    原文 你希望保持你的controller足够简单. 你的controller越来越臃肿,你听说command模式是一个给controller瘦身的解决方案. 但是你不知道command模式是否适合你的 ...

  6. [译]A NON-TRIVIAL EXAMPLE OF MEDIATR USAGE

    原文 来看看我目前的一个项目.这个是一个多租户的财务跟踪系统.有一个组织继承的关系.首先得新建一个组织. 表单如下: 这个表单能让用户输入关于组织的一些信息,包括active directory组,一 ...

  7. 【译】Dependency Injection with Autofac

    先说下为什么翻译这篇文章,既定的方向是架构,然后为了学习架构就去学习一些架构模式.设计思想. 突然有一天发现依赖注入这种技能.为了使得架构可测试.易维护.可扩展,需要架构设计为松耦合类型,简单的说也就 ...

  8. Ninject.Extensions.

    最近在使用IoC进行一个较复杂的项目进行架构,在IoC的选择上让我很是纠结.首先我不喜欢大量的配置文件进行配置,那简直是噩梦,比学习一门编程语言还痛苦.我喜欢前一段时间看EF的CodeFirst的那种 ...

  9. 为什么使用Ninject?

    Ninject 3 学习笔记 一.为什么使用Ninject? 分类: 程序2012-11-10 19:23 2209人阅读 评论(0) 收藏 举报 c#iocNinject框架注入 最近在使用IoC进 ...

随机推荐

  1. Age of Moyu HDU - 6386 (杭电多校7A)

    给出n和点,m条边,每条边有各自的标号,进入第一个标号需要消耗1的费用,此后转换标号需要1费用,在同一个标号上走不需要费用.问你从1到n最少需要多少费用. 最短路变形,把第一个点看成不存在的标号,然后 ...

  2. Glad You Came hdu-6356(ST表 || 线段树)

    第一种用线段树,用两颗数维护区间最大值和区间的最小值,然后更新的时候如果我目前区间内的最大值比我得到的v小,那么我就把这个区间修改成v,如果我的最小值比v大,那么v就是没有用的,直接跳过,然后这样每次 ...

  3. urls 管理

    问题阐述:如何管理多个app下的路由分发,使得管理更加清晰? 1. 在app下创建urls.py文件 from django.conf.urls import url from django.urls ...

  4. The 2018 ACM-ICPC Asia Qingdao Regional Contest

    The 2018 ACM-ICPC Asia Qingdao Regional Contest 青岛总体来说只会3题 C #include<bits/stdc++.h> using nam ...

  5. Spring 声明式事务

    事务的特性/概念 事务:一组操作要么都成功要么失败: 事务的四个关键属性(ACID): 原子性(atomicity):“原子”的本意是“不可再分”,事务的原子性表现为一个事务中涉及到的多个操作在逻辑上 ...

  6. spring activemq 整合

    创建maven项目 项目目录结构为 首先配置相关maven依赖 <!-- 版本管理 --> <properties> <springframework>4.1.8. ...

  7. 【CSS 技能提升】 :before和:after的使用

    前几天的晚上较全面的去看了下css的一些文档和资料,大部分的样式运用都没什么大问题了,只是有些许较陌生,但是也知道他们的存在和实现的是什么样式.今天主要想在这篇学习笔记中写的也不多,主要是针对:bef ...

  8. 记录EXCEL格式和TXT文本格式之间的互转

    EXCEL格式转变成TXT文本格式 1.打开execl文档,点击文件另存为 2.选择txt保存 3.重命名文档,打开该txt文档 4.按Ctrl+H,将文档中空格转换成其他分割符,单击确定 TXT格式 ...

  9. 拆分字符串法 获取url的GET参数

    function serilizeURL(url){ var rs=url.split("?")[1]; var arr=rs.split("&"); ...

  10. 关于shared_ptr与weak_ptr的使用(good)

    shared_ptr是带引用计数的智能指针,可以说大部分的情形选择用shared_ptr不会出问题.那么weak_ptr是什么,应该怎么用呢? weak_ptr也是智能指针,但是比较弱,感觉没什么用. ...