Redis 消息中间件 ServiceStack.Redis 轻量级
问题:
公司开了个新项目,算上我一共3个人。车间里机台通过流水线连通联动的玩意。一个管理控制系统连接各个机台和硬件。专机类型就有5种,个数差不多20个左右。
软件规划的时候采用总分的结构,管理控制系统和专机子系统之间通过消息中间件通讯。本来也想TCP连接来着,但是开发时间不允许。而且每个系统都得写一遍这个玩意。
消息中间件有很多个,比如 Kafka、RabbitMQ、RocketMQ等国内外的消息中间件。这些中间件无论宣称的多么轻量级都要啃一下,更要命的是就他娘三个人。而且后面还要这个鸡儿系统可复制。
考虑到消息及时性、开发难易程度、维护简便性等因素后决定用Redis的pub/sub功能来实现.软件结构大概如类似结构。
可用性:
作为消息通知属于安装了Redis就有的功能,因为Redis是用在系统中存储一些热数据,不用单独维护,在Windows中属于服务直接就开了。
作为可以分布式集群使用的数据库,消息传递应该比较OK了。虽然使用的client-server,但是server-server已经很好了。料想client-server也不会差
试验消息内容发送订阅的情况下,速度在30毫秒内,貌似可以。看其他博主说大于10K入队比较慢,但是可以不用入消息队列啊,用发布订阅。
.net 下一般使用ServiceStack.Redis,要命的是4.0以后收费,可以破解的但是不支持List<T>型的数据直接存取,想用只能变成JSON字符串存着。
- 如果只是用订阅发布功能,不存储热数据或者不使用List<T>的数据可以使用4.0以上的版本。文末会贴上两个类型的下载包。想用其他的包也可以,我这里只说一种思路。
实现:
模块结构图展示如下
- public static class MSServer
- {
- // 定义一个object对象
- private static object objinstance = new object();
- private static ServerState CurState = ServerState.Free;
- static PooledRedisClientManager prcm;
- private static string clientmake = string.Empty;
- /// <summary>
- /// 连接的地址
- /// </summary>
- /// <param name="IP">地址127.0.0.1:6379</param>
- /// <param name="rechannels">接收通道 {"channel:1-13","channel:1-5"}</param>
- /// <returns></returns>
- public static int OpenServer(string IP ,string[] rechannels)
- {
- try
- {
- if (prcm == null)
- {
- lock (objinstance)
- {
- if (prcm == null)
- {
- prcm = CreateManager(IP, IP);
- CurState = ServerState.Init;
- return CreateLink(rechannels);
- }
- }
- }
- }
- catch
- {
- prcm = null;
- CurState = ServerState.Free;
- return -;
- }
- return ;
- }
- private static int CreateLink(string[] SourceID)
- {
- if (CurState == ServerState.Init && SourceID.Length > )
- {
- try
- {
- using (IRedisClient Redis = prcm.GetReadOnlyClient())
- {
- clientmake = SourceID[];
- var info = Redis.GetClientsInfo().Where(i => i["name"] == clientmake).ToList();
- info.ForEach(i =>
- {
- Redis.KillClient(i["addr"]);
- });
- Redis.SetClient(clientmake);
- IRedisSubscription sc = Redis.CreateSubscription();
- Task.Run(() =>
- {
- try
- {
- sc.SubscribeToChannels(SourceID);
- }
- catch { }
- });
- sc.OnMessage += new Action<string, string>(showpub);
- }
- CurState = ServerState.Work;
- }
- catch
- {
- string message = string.Empty;
- prcm = null;
- CurState = ServerState.Free;
- return -;
- }
- return ;
- }
- else
- {
- return ;
- }
- }
- public static Action<string, string> ReceiveMessage;
- static void showpub(string channel, string message)
- {
- if (ReceiveMessage != null)
- {
- ReceiveMessage(channel, message);
- }
- }
- private static PooledRedisClientManager CreateManager(string writeHost, string readHost)
- {
- var redisClientConfig = new RedisClientManagerConfig
- {
- MaxWritePoolSize = ,//“写”链接池链接数
- MaxReadPoolSize = ,//“读”链接池链接数
- DefaultDb = ,
- AutoStart = true,
- };
- //读的客户端只能接受特定的命令,不能用于发送信息
- var RedisClientManager = new PooledRedisClientManager(
- new string[] { writeHost }//用于写
- , new string[] { readHost }//用于读
- , redisClientConfig);
- CurState = ServerState.Init;
- return RedisClientManager;
- }
- /// <summary>
- /// 发送信息
- /// </summary>
- /// <param name="channel">通讯对象 "channel:1-13"</param>
- /// <param name="meesage">发送信息 "test send "</param>
- /// <returns>0 发送失败 1 发送成功 -1 连接损毁 检查网络后重建</returns>
- public static long PubMessage(string channel, string meesage)
- {
- if (CurState == ServerState.Work)
- {
- if (!string.IsNullOrEmpty(channel) && !string.IsNullOrEmpty(meesage))
- {
- try
- {
- using (IRedisClient Redis = prcm.GetClient())
- {
- Redis.SetClient(clientmake);
- return Redis.PublishMessage(channel, meesage);
- }
- }
- catch
- {
- prcm = null;
- CurState = ServerState.Free;
- return -;
- }
- }
- else
- {
- return ;
- }
- }
- else
- {
- return -;
- }
- }
- }
- public enum ServerState
- {
- Free,
- Init,
- Work,
- Del
- }
有一个问题,就是连接远程的服务器时如果网络断开再重连,会残留没用的client ,这样如果网络断断续续的话,会留好多没有清除的客户端。
这个在3.0.504版本中Redis 中也有这个问题,不知道是基于什么考虑的。所以需要建立连接的时候,给个客户端名称,再初始化的时候删掉所有同类型的名称。
使用的时候大概类似操作 textbox2.text = "channel:1-5" .为了简便发布的和监听的都是本地的一个通道。
- private void button1_Click(object sender, EventArgs e)
- {
- //11.1.7.152 192.168.12.173
- int result = ServerMS.MSServer.OpenServer("127.0.0.1:6379", new string[] { textBox2.Text });
- label1.Text = result.ToString();
- //1匿名事件
- ServerMS.MSServer.ReceiveMessage += new Action<string, string>(fuck);
- if (result == )
- {
- //发送失败重新发送 检查 通道和字符串后重新发送
- }
- else if (result == )
- {
- //发送成功
- }
- else if (result == -)
- {
- //连接错误 需要 ServerMS.MSServer.OpenServer("192.168.12.173:6379", new string[] { textBox2.Text });
- }
- }
- void fuck(string channel, string message)
- {
- this.BeginInvoke(new Action(() =>
- {
- textBox4.Text = channel + message;
- }));
- }
- public bool sdfsd = true;
- private void button3_Click(object sender, EventArgs e)
- {long result = ServerMS.MSServer.PubMessage(textBox2.Text, DateTime.Now.ToString("yyyyyMMddhhmmssfff"));
- if (result == )
- {
- //发送失败重新发送
- }
- else if (result == )
- {
- //发送成功
- }
- else if (result == -)
- {
- //连接错误 需要 ServerMS.MSServer.OpenServer("192.168.12.173:6379", new string[] { textBox2.Text });
- }
- }
为了简便channel:是通道的固定命令 ,可以自定义channel:后面的内容,发送就有反馈。确保所有机台都接收到。
如果有断线的需要程序自己重连,接收通道的客户端不可以再给其他的使用,Redis上说Redis client 进入订阅模式时只能接受订阅发布等命令指令,不接受普通的存取和其他命令
所以如果需要在读取、写入、发布、执行其他的指令需要使用其他客户端,否则就出错了。跑了几天了上亿次的测试貌似没有出现什么问题。
发布订阅消息不会走AOF RDB只存在于内存中,即发即用,用完就没了。没在线就没了。需要考虑使用环境。
还用ping pong来确定连接状态,也可以自定义数据,使用场景要自己开发,要适合自己的才是好的。
下载:
4.0 dll
链接:https://pan.baidu.com/s/1966t0pduHxQXcxcxV3ZTeQ
提取码:js8p
5.8 dll不可以使用List<T>类型
链接:https://pan.baidu.com/s/1RFgY4V0ZO78Wvd7LOxr97g
提取码:bxh2
Redis 消息中间件 ServiceStack.Redis 轻量级的更多相关文章
- Redis客户端ServiceStack.Redis的简单使用
在nuget中下载ServiceStack.Redis,但是运行之后会出现一个问题: Exception: "Com.JinYiWei.Cache.RedisHelper"的类型初 ...
- 关于 ServiceStack.Redis 4.0 License
今天更新了框架中的Redis驱动ServiceStack.Redis,最新版本4.0.5.0. 在做简单压力测试时出现异常,提示每小时允许6000个请求. The free-quota limit o ...
- 『性能』ServiceStack.Redis 和 StackExchange.Redis 性能比较
背景 近来,需要用到 Redis 这类缓存技术 —— MongoDB 和 Redis 没有进行过比较. 我也懒得在这些细节上 纠结那么多 —— 按照网友给出的文章,听从网友建议,选择 Redis. R ...
- servicestack.redis工具类
using System;using System.Collections.Generic;using System.Linq;using ServiceStack.Redis;using Servi ...
- 使用ServiceStack.Redis实现Redis数据读写
原文:使用ServiceStack.Redis实现Redis数据读写 User.cs实体类 public class User { public string Name { get; set; } p ...
- .Net使用Redis详解之ServiceStack.Redis(七)
序言 本篇从.Net如何接入Reis开始,直至.Net对Redis的各种操作,为了方便学习与做为文档的查看,我做一遍注释展现,其中会对list的阻塞功能和事务的运用做二个案例,进行记录学习. Redi ...
- ServiceStack.Redis订阅发布服务的调用(Z)
1.Redis订阅发布介绍Redis订阅发布是一种消息通信模式:发布者(publisher)发送消息,订阅者(Subscriber)接受消息.类似于设计模式中的观察者模式.发布者和订阅者之间使用频 ...
- serviceStack.Redis 在PooledRedisClientManager 中设置密码
ServiceStack.Redis 是一个C#访问Redis的客户端,可以说可以通过它实现所有需要Redis-Cli的功能.但是今天我在主Redis 实例设置了访问密码,而在slave 上没有设置, ...
- ServiceStack.Redis订阅发布服务的调用
1.Redis订阅发布介绍 Redis订阅发布是一种消息通信模式:发布者(publisher)发送消息,订阅者(Subscriber)接受消息.类似于设计模式中的观察者模式. 发布者和订阅者之间使用频 ...
随机推荐
- 素数&欧拉函数
素数表 const int maxN找[1,maxN)内的素数 int prime[int I]第I个素数 const int maxN=1e5+5; int prime[maxN]; bool ma ...
- dispatch_async 的 block 中是否该使用_weak self
问题分析 我看过很多文章关于在dispatch_async的block里面使用_weak self, 但是让我疑惑的是,以下代码是否需要必须使用_weak self, 因为我也看到了很多观点说,在有些 ...
- Golang项目部署
文章来源:https://goframe.org/deploymen... 一.独立部署 使用GF开发的应用程序可以独立地部署到服务器上,设置为后台守护进程运行即可.这种模式常用在简单的API服务项目 ...
- mac OS nvm 常用命令
nvm install stable ## 安装最新稳定版 node,当前是node v10.15.0 (npm v6.4.1) nvm install <version> ## 安装指定 ...
- OSChina 清明节乱弹 ——准备好纸巾了看乱弹
2019独角兽企业重金招聘Python工程师标准>>> Osc乱弹歌单(2017)请戳(这里) [今日歌曲] @亚麻仔 :分享 范忆堂 的歌曲<酡颜 (夏热版)> 同一个 ...
- 使用 html5 FileReader 获取图片, 并异步上传到服务器 (不使用 iframe)
为什么80%的码农都做不了架构师?>>> 原理: 1.使用FileReader 读取图片的base64编码 2.使用ajax,把图片的base64编码post到服务器. 3.根据 ...
- pod setup命令失败解决方法
最近运行pod setup出现以下问题: remote: Compressing objects: 100% (34/34), done.error: RPC failed; curl 56 SSLR ...
- Netty(一):ByteBuf读写过程图解
我们知道ByteBuf通过读写两个索引分离,避免了NIO中ByteBuffer中读写模式切换时,需要flip等繁琐的操作. 今天就通过一段测试代码以及图例来直观的了解下ByteBuf中的readInd ...
- OpenStack 删除instance 和其附加的volumes
在openstack里面有时候删除instance时,volume无法跟着删除,可以自己编写脚本来实现, 脚本代码如下: #!/bin/bash for i in $(cat /root/host-d ...
- Vue.js 条件渲染 v-if、v-show、v-else
v-if v-if 完全根据表达式的值在DOM中生成或移除一个元素.如果v-if表达式赋值为false,那么对应的元素就会从DOM中移除:否则,对应元素的一个克隆将被重新插入DOM中. 1 2 3 ...