MediatR 中介模式
使用MediatR完成基于内存级别的消息发布订阅
在微服务架构中领域驱动模型中处理领域事件的相关操作
在区分好领域模型后,就拿代码中来说嘛,用户领域中添加用户操作可能或存在跟用户相关的一些领域事件,在添加用户的时候会执行相关的领域事件
首先需要添加nuget包 MediatR
MediatR中有几个对象:IRequest,IRequestHandler,INotificationHandler
具体用法可以看下MediatR github
领域模型需要继承Entity,里面封装了 领域事件的相关操作
public abstract class Entity
{
int? _requestedHashCode;
int _Id;
public virtual int Id
{
get
{
return _Id;
}
protected set
{
_Id = value;
}
} private List<INotification> _domainEvents;
public IReadOnlyCollection<INotification> DomainEvents => _domainEvents?.AsReadOnly(); public void AddDomainEvent(INotification eventItem)
{
_domainEvents = _domainEvents ?? new List<INotification>();
_domainEvents.Add(eventItem);
} public void RemoveDomainEvent(INotification eventItem)
{
_domainEvents?.Remove(eventItem);
} public void ClearDomainEvents()
{
_domainEvents?.Clear();
} public bool IsTransient()
{
return this.Id == default(Int32);
} public override bool Equals(object obj)
{
if (obj == null || !(obj is Entity))
return false; if (Object.ReferenceEquals(this, obj))
return true; if (this.GetType() != obj.GetType())
return false; Entity item = (Entity)obj; if (item.IsTransient() || this.IsTransient())
return false;
else
return item.Id == this.Id;
} public override int GetHashCode()
{
if (!IsTransient())
{
if (!_requestedHashCode.HasValue)
_requestedHashCode = this.Id.GetHashCode() ^ ; return _requestedHashCode.Value;
}
else
return base.GetHashCode(); }
public static bool operator ==(Entity left, Entity right)
{
if (Object.Equals(left, null))
return (Object.Equals(right, null)) ? true : false;
else
return left.Equals(right);
} public static bool operator !=(Entity left, Entity right)
{
return !(left == right);
}
}
Entity
如:添加用户的时候添加一个默认角色相关的表处理
public class TbUser : Entity, IAggregateRoot
{ public string UserName { get; set; }
public string UserPasswrod { get; set; }
public string UserPhone { get; set; }
public string XXXXX { get; set; }
[NotMapped]
public List<TbUserRole> TbUserRoles { get; set; }
public TbUser()
{
TbUserRoles = new List<TbUserRole>();
AddDomainEvent(new UserCreatedEvent() { tbUser = this });
} #region 领域职责 结合领域事件处理 黎又铭
/*
如:添加用户的时候添加默认角色 其实本身是可以直接在业务逻辑中写的
有点类是SQL触发器,添加用户的时候构建件默认角色信息模型触发事件添加角色信息 这里有注意到 实体模型并没有强制处理表接口的主外键关系 只是做了模型对象映射
*/
/// <summary>
/// 添加用户职责 只构建职责 不处理 处理交友领取事件的Command CommandHandler处理
/// </summary>
public void AddDefalutRole(TbUserRole tbUserRole)
{
TbUserRoles.Add(tbUserRole);
//添加默认角色的领域事件
AddDomainEvent(new UserRoleCreatedEvent() { TbUserRole = tbUserRole }); } #endregion }
User
先定义一个创建用户的命令操作: 这里的TbUser是领域时间模型,传递数据以及对领域模型事件操作
public class UserCreateCommand : IRequest<TbUser>
{ public TbUser tbUser{ get; set; }
}
public class UserCreateCommandHandler : IRequestHandler<UserCreateCommand, TbUser>
{ private IUserRepository _userRepository; public UserCreateCommandHandler(IUserRepository userRepository)
{
_userRepository = userRepository;
} public async Task<TbUser> Handle(UserCreateCommand request, CancellationToken cancellationToken)
{ await _userRepository.AddUserAsync(request.tbUser);
await _userRepository.UnitOfWork.SaveEntitiesAsync();
return request.tbUser;
}
}
这里还需要一个MediatR的扩展,发布领域事件
static class MediatorExtension
{
/// <summary>
/// 异步处理领域事件
/// </summary>
/// <param name="mediator"></param>
/// <param name="ctx"></param>
/// <returns></returns>
public static async Task DispatchDomainEventsAsync(this IMediator mediator, UserDbContext ctx)
{
var domainEntities = ctx.ChangeTracker
.Entries<Entity>()
.Where(x => x.Entity.DomainEvents != null && x.Entity.DomainEvents.Any()); var domainEvents = domainEntities
.SelectMany(x => x.Entity.DomainEvents)
.ToList(); domainEntities.ToList()
.ForEach(entity => entity.Entity.ClearDomainEvents()); var tasks = domainEvents
.Select(async (domainEvent) => {
await mediator.Publish(domainEvent);
}); await Task.WhenAll(tasks);
}
}
扩展
下面是订阅领域事件
public class UserRoleNotification : INotificationHandler<UserRoleCreatedEvent>
{
private IUserRepository _userRepository;
public UserRoleNotification(IUserRepository userRepository)
{
_userRepository = userRepository;
}
public async Task Handle(UserRoleCreatedEvent notification, CancellationToken cancellationToken)
{ await _userRepository.AddUserRoleAsync(notification.TbUserRole);
}
在controller中的使用:
[HttpPost]
[Route("adduser")]
public async Task<IActionResult> AddUser([FromBody] TbUser tbUser)
{
tbUser.AddDefalutRole(new TbUserRole { RoleId = , UserId = });
var command = new UserCreateCommand() { tbUser = tbUser };
var result = await _mediator.Send(command);
return Ok(result);
}
看下效果

MediatR 中介模式的更多相关文章
- [工作中的设计模式]中介模式模式Mediator
一.模式解析 用一个中介者对象封装一系列的对象交互,中介者使各对象不需要显示地相互作用,从而使耦合松散,而且可以独立地改变它们之间的交互. 中介模式又叫调停者模式,他有如下特点: 1.有多个系统或者对 ...
- 中介模式和学习日记Effective C++
中介模式(Mediator):利用中介对象来封装一组对象交互.中保使对象并不需要显式地相互引用,使得松耦合,的交互. (1).中介者模式非常easy在系统中应用,也非常easy在系统中误用.当系统出现 ...
- Java设计模式(16)中介模式(Mediator模式)
Mediator定义:用一个中介对象来封装一系列关于对象交互行为. 为何使用Mediator模式/中介模式 各个对象之间的交互操作非常多,每个对象的行为操作都依赖彼此对方,修改一个对象的行为,同时会涉 ...
- python 设计模式之中介模式
Mediator Pattern:中介模式 中介模式提供了一系列统一的系统接口.此模式也被认为是行为模式,因为他能选择程序处理流程. 当许多类开始在交互中产生结果时,可以选用中介模式.当软件开始组织 ...
- .NET Core 使用MediatR CQRS模式
前言 CQRS(Command Query Responsibility Segregation)命令查询职责分离模式,它主要从我们业务系统中进行分离出我们(Command 增.删.改)和(Query ...
- .NET Core 使用MediatR CQRS模式 读写分离
前言 CQRS(Command Query Responsibility Segregation)命令查询职责分离模式,它主要从我们业务系统中进行分离出我们(Command 增.删.改)和(Query ...
- <人人都懂设计模式>-中介模式
真正的用房屋中介来作例子, 好的书籍总是让人记忆深刻. class HouseInfo: def __init__(self, area, price, has_window, has_bathroo ...
- Github上优秀的.NET Core项目
Github上优秀的.NET Core开源项目的集合.内容包括:库.工具.框架.模板引擎.身份认证.数据库.ORM框架.图片处理.文本处理.机器学习.日志.代码分析.教程等. Github地址:htt ...
- 【转载】Github上优秀的.NET Core项目
Github上优秀的.NET Core项目 Github上优秀的.NET Core开源项目的集合.内容包括:库.工具.框架.模板引擎.身份认证.数据库.ORM框架.图片处理.文本处理.机器学习.日志. ...
随机推荐
- 一本通1622Goldbach’s Conjecture
1622:Goldbach’s Conjecture 时间限制: 1000 ms 内存限制: 524288 KB [题目描述] 原题来自:Ulm Local,题面详见:POJ 2262 ...
- BZOJ2303 APIO2011方格染色(并查集)
比较难想到的是将题目中的要求看做异或.那么有ai,j^ai+1,j^ai,j+1^ai+1,j+1=1.瞎化一化可以大胆猜想得到a1,1^a1,j^ai,1^ai,j=(i-1)*(j-1)& ...
- C#委托的介绍(delegate、Action、Func、predicate)ga
转载:http://www.cnblogs.com/akwwl/p/3232679.html 感觉写的很好.例子也很简单明了.赞一个 委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参 ...
- 【刷题】LOJ 6225 「网络流 24 题」火星探险问题
题目描述 火星探险队的登陆舱将在火星表面着陆,登陆舱内有多部障碍物探测车. 登陆舱着陆后,探测车将离开登陆舱向先期到达的传送器方向移动. 探测车在移动中还必须采集岩石标本. 每一块岩石标本由最先遇到它 ...
- 解决telnet不是内部命令
1.telnet在win7下默认是不开启的,所以需要我们自己手动开启.那么首先我们点击开始菜单,找到控制面板项,点击进入: 2.进入程序和功能模块,我们在左边需要选择“打开或关闭windows功能”, ...
- Graham's Scan法求解凸包问题
概念 凸包(Convex Hull)是一个计算几何(图形学)中的概念.用不严谨的话来讲,给定二维平面上的点集,凸包就是将最外层的点连接起来构成的凸多边型,它能包含点集中所有点的.严谨的定义和相关概念参 ...
- MathExam V2.0
# 隔壁村小学的小朋友都羡慕哭了2.0版 一.预估与实际 PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟) Planning 计划 1 ...
- css3 :nth-child()选择器的使用
一.nth-child() 1. nth-child(n):父元素下的第n个子元素,n>0,索引从1开始.不区分子元素类型. 2. nth-child(odd):父元素下的奇数子元素,等同于:n ...
- Spark记录-Scala数组
Scala提供了一种数据结构叫作数组,数组是一种存储了相同类型元素的固定大小顺序集合.数组用于存储数据集合,但将数组视为相同类型变量的集合通常更为有用. 可以声明一个数组变量,例如:numbers,使 ...
- Java 在匿名内部函数中为外部函数变量赋值的解决方案
Java匿名内部函数本人不怎么主动使用,但是经常会调用一些API,其中会调用一些接口,而这些接口是需要使用匿名内部类来实现的,于是就遇到了一些问题. 就比如okHttp3 的接口调用 OkHttpCl ...