前言

  “一旦开始了就要坚持下去“。为什么本文的第一句话是这么一句话呢,因为我经常就是开头轰轰烈烈,结果越来越枯燥,就不想做下去了。但是版图就放弃又那么不甘心,继续加油吧。

  吐槽完毕,进入正题。在上一篇中我们的主角LayIM已经登场了。而且界面已经实现,那么有些小伙伴就有疑惑了,详细流程是什么样的,今天我就介绍一个东西,那就是 /layim/init 接口的实现细节

  另外,项目已经升级至 .NET CORE 2.1 ,最新代码在 dev-netcore2.1 分支上

需求

  需求是什么?需求是将LayIM的主界面显示出来。如下:

   

  嗯,就是这么个东西。在上一篇中我已经介绍到,init接口就是为了实现主界面的,那么主界面的数据是什么样的呢?(其实在两年前的博客中已经讲过,这里在稍微提一下)我们看一下DEMO中的getList.json.

  

  如上图,我们只要定义一个接口然后输出上图格式中的json数据即可。由于我在框架中用SQLServer给了一个默认实现,关系型数据库那肯定是各种关系啦。首先定义最基本的表。

  用户表,用户好友分组表,用户好友关系表,群组表,群组群员关系表。目前这五个表足够了。

  下面逐个分析击破:

  mine:当前用户信息,从用户表取出即可

  friend:首先从用户好友分组表取出当前用户所有的分组,在从用户好友关系表取出好友。然后程序中去匹配该好友对应的分组即可。及分组和好友是一对多的关系

  group:从群组群员关系表中取出当前用户所在的群,然后在从群组表取出相应的群信息,组合数据即可

具体代码实现

  下面的代码如果有不理解的地方,可以先从前面几篇中补补课。

  首先,毋庸置疑,先将路由配置上:

  routes.AddQuery("/init", async context => await GetInitData(context));

  GetInitData 做了什么事呢?它就是获取注册的 ILayIMStorage 实例,然后调用接口方法获取数据。

   public interface ILayIMStorage
{
/// <summary>
/// 初始化数据
/// </summary>
/// <param name="userId"></param>
/// <returns></returns>
Task<LayIMInitModel> GetInitData(string userId); /// <summary>
/// 保存聊天记录
/// </summary>
/// <param name="message"></param>
/// <returns></returns>
Task<int> SaveMessage(LayIMMessageModel message);
}

接口实现类

  在项目 LayIM.AspNetCore.Storage.SqlServer中实现ILayIMStorage接口.Repository具体代码就不在粘贴了,就是一系列Sql的查询然后数据组合。

        /// <summary>
/// 获取初始化数据
/// </summary>
/// <param name="userId"></param>
/// <returns></returns>
public async Task<LayIMInitModel> GetInitData(string userId)
{
var mineTsk = userRepository.GetUserById(userId);
var friendGroupsTask = friendGroupRepository.GetUserGroups(userId);
var friendRelationsTask = friendRelationRepository.GetFriendRelations(userId);
var groupIdsTask = groupMemberRepository.GetUserBigGroups(userId); LayIMInitModel initModel = new LayIMInitModel
{
//用户自己
mine = await mineTsk
};
//好友列表
List<FriendGroupModel> friend = new List<FriendGroupModel>(); IEnumerable<FriendGroupModel> friendGroups = await friendGroupsTask;
IEnumerable<FriendRelationShip> friendRelations = await friendRelationsTask;
IEnumerable<long> friendIds = friendRelations.Select(x => x.FriendId); IEnumerable<UserModel> friends = await userRepository.GetUsersByIds(friendIds); if (friendIds?.Count() > )
{
foreach (var group in friendGroups)
{
var friendIdsInGroup = friendRelations.Where(r => r.GroupId == group.id).Select(r => r.FriendId); group.list = friends.Where(x => friendIdsInGroup.Any(f => f == x.id));
}
} friend.AddRange(friendGroups); initModel.friend = friend; //群组列表
IEnumerable<long> groupIds = await groupIdsTask;
var bigGroupsTask = bigGroupRepository.GetBigGroups(groupIds);
initModel.group = await bigGroupsTask; return initModel;
}

  写完方法之后,记得要注册服务。

        /// <summary>
/// 使用SqlServer
/// </summary>
/// <param name="services"></param>
/// <param name="setConfig"></param>
public static IServiceCollection AddSqlServer(this IServiceCollection services,string connectionString)
{
var dbConfig = new DBConnectionConfig(DBType.SqlServer)
{
ConnectionString = connectionString
};
services.AddSingleton(dbConfig);
services.AddSingleton<ILayIMStorage, LayIMDapperStorage>();
return services;
}

  然后在Demo中的使用方法如下:

 services.AddLayIM().AddRongCloud(config =>
{
config.AppKey = "appkey";
config.AppSecret = "appsecret";
})
.AddSqlServer("connectionString");

  重启项目,测试一下 layim/init 接口。数据返回正常

  

  当然,一个好友的用户数据一般变化不大,所以对于接口要增加缓存设计。不过这里我暂时没有实现,只是先临时用了 IMemoryCache

请求回放

  首先,由于请求的 Path /layim 开头的,所以中间件判定为LayIM请求,然后在经过路由转发器,找到对应的处理器。

 var dispatcher = LayIMRoutes.Routes.FindDispatcher(path);

  之前已经介绍过,Dispatcher 是实现了Dispatch 方法的。

 internal interface ILayIMDispatcher
{
Task Dispatch(HttpContext context);
}

  所以,最终的请求其实都会落到路由注册的方法中,也就是上文中的  routes.AddQuery("/init", async context => await GetInitData(context)); 那么其他的返回ContentType,转JSON就是框架实现了,业务没必要关心它了。

  说到这里可能大家还是比较迷糊,下面是一个Dispatcher之间的关系图.

  

  MethodFilterDispatcher 抽象类 负责请求方法的校验。GET POST PUT DELETE 等

  CommandDispatcher<TResult> 抽象类 负责主业务逻辑处理,模板方法,处理细节交给实现类

  ExecuteCommandDispatcher<TResult> 实现类,POST请求

  QueryCommandDispatcher<TResult> 实现类,GET 请求

  其他待扩展

总结

  本文通过一个 /layim/init 接口的详细介绍,通过这个接口请求流程,大家能够对框架的处理部分设计有所理解。另外,本文内容都是在后端,至于前台怎么绑定数据处理等,咱们下回分解。

  博客预告:LayIM.AspNetCore Middleware 开发日记(六)嵌入资源的使用,layim.config的封装

  项目地址:https://github.com/fanpan26/LayIM.AspNetCore  欢迎小伙伴们star 围观 提意见。

LayIM.AspNetCore Middleware 开发日记(五)Init接口实现细节的更多相关文章

  1. LayIM.AspNetCore Middleware 开发日记(一)闲言碎语

    前言 前几天写博客的时候突然看见了历史上的今天.不禁感慨时光如梭,这系列博客后来被我标注了已经过时,但是还有很多小伙伴咨询我.既然过时就要更新,正好 .NET Core 也出来很久了,于是乎想到把La ...

  2. LayIM.AspNetCore Middleware 开发日记(四)主角登场(LayIM介绍)

    前言 在前几篇中已经初步介绍了开发AspNetCore中间件的一些基础知识,不过都没有很深入的去研究,后续还是需要去看看源码.本篇呢,终于有点开头的味道了,就是要介绍LayIM了,其实标题写的是主角, ...

  3. LayIM.AspNetCore Middleware 开发日记(三)基础框架搭建

    前言 在上一篇中简单讲了一些基础知识,例如Asp.Net Core Middleware 的使用,DI的简单使用以及嵌入式资源的使用方法等.本篇就是结合基础知识来构建一个基础框架出来. 那么框架有什么 ...

  4. LayIM.AspNetCore Middleware 开发日记(六)嵌入资源的使用,layim.config的封装

    前言 距离上一篇博客竟然已经10多天了...工作上的事,个人原因,种种吧.不多说废话,本文将会重点介绍layim的入口配置. LayIM配置 其实在开发者文档里面已经描述的很清楚了.除了几个重要的接口 ...

  5. LayIM.AspNetCore Middleware 开发日记(二)预备知识介绍

    前言 开发一个AspNetCore的中间件需要理解RequestDelegate.另外,还需要理解.NET Core中的依赖注入.还有一个就是内嵌资源的访问.例如:EmbeddedFileProvid ...

  6. LayIM.AspNetCore Middleware 开发日记(七)Asp.Net.Core.SignalR闪亮登场

    前言   前几篇介绍了整个中间件的构成,路由,基本配置等等.基本上没有涉及到通讯部分.不过已经实现了融云的通讯功能,由于是第三方的就不在单独去写.正好.NET Core SignalR已经出来好久了, ...

  7. 【Android开发日记】之入门篇(五)——Android四大组件之Service

    这几天忙着驾校考试,连电脑都碰不到了,今天总算告一段落了~~Service作为Android的服务组件,默默地在后台为整个程序服务,辅助应用与系统中的其他组件或系统服务进行沟通.它跟Activity的 ...

  8. 五、利用EnterpriseFrameWork快速开发基于WebServices的接口

    回<[开源]EnterpriseFrameWork框架系列文章索引> EnterpriseFrameWork框架实例源代码下载: 实例下载 前面几章已完成EnterpriseFrameWo ...

  9. 嵌入式Linux驱动开发日记

    嵌入式Linux驱动开发日记 主机硬件环境 开发机:虚拟机Ubuntu12.04 内存: 1G 硬盘:80GB 目标板硬件环境 CPU: SP5V210 (开发板:QT210) SDRAM: 512M ...

随机推荐

  1. 解决-word里无论怎么改变字体颜色,字体总是红色的

    1.你遇到的问题是Word当前处于审阅状态,修改的内容显示为红色字体.2.解决办法是退出Word审阅状态,或者接受全部修订.3.不同Word版本的审阅模式不同,可在菜单栏里退出审阅,或者按鼠标右键弹出 ...

  2. ROS:消息发布器和订阅器(c++)

    学习资料主要源自http://wiki.ros.org/ROS/Tutorials/WritingPublisherSubscriber%28c%2B%2B%29 $ roscd beginner_t ...

  3. 领域模型(DomainModel)与视图模型(ViewModel)

    Model-View-Controller(模型-视图-控制器,MVC)模式将你的软件组织并分解成三个截然不同的角色: Model 封装了你的应用数据.应用流程和业务逻辑. View 从 Model ...

  4. js-js的运算

    ** js里面不区分整数和小数 var j = 123; alert(j/1000*1000); //在Java里面结果是0 //在js里面不区分整数和小数 123/1000 = 0.123 *100 ...

  5. COCI2017-2018-2 San

    题意 有\(n \leq 40\)个节点,每个节点有权值\(H \leq 1e9\)和贡献\(v \leq 1e9\),从任意一个点可以向右跳到一个权值不小于它的节点,并获得该点贡献 可以从任意一个点 ...

  6. 【python基础】之str类字符串

    str类字符串是不可变对象 1.创建字符串 s1 = str() #创建一个空字符串 s2 = str("hello") #创建字符串"hello" 2.处理字 ...

  7. .NET开源工作流RoadFlow-流程设计-流程步骤设置-事件设置

    事件设置是设置当前步骤在提交前后或退回前后要执行的一些操作(该事件为服务器事件). 事件格式为:dll名称.命名空间名称.类名.方法名,这里不需要写括号和参数,处理时会自动带上当前流程实例的相关参数. ...

  8. Codeforces Round #419 A+B

    A. Karen and Morning time limit per test  2 seconds memory limit per test  512 megabytes   Karen is ...

  9. IOS xcode安装

    xcode软件下载地址: 可以通过虚拟机共享文件夹将xcode传递给虚拟机上的os系统: 第一个程序创建:

  10. PHP中die()、exit()、return的区别

    1.die()是exit()的别名函数,用法与exit()完全相同,php手册上的说法是,“使用这种别名通常不是个好主意,因为这种别名可能会被完全废弃或更名,导致脚本难以移植到新版本的 PHP 上.” ...