Redis系列(二)-Hredis客户端设计及开源
接上篇c#实现redis客户端(一),重新整理些了下。
阅读目录:
项目说明
背景:因为有地方要用,而又没找到对sentinel良好支持的Net客户端,所以就简单重写了个。
目标:尽可能的简单,轻量级,不进行过度的封装,使用方便。
代码说明:
1. 与Redis服务端的Socket通信、协议格式封装。在RedisBaseClient里
2. 只对Set、Get封装,暴露出Send接口。在RedisCommand里面添加自己想要的支持。
var info = rcClient.Send(RedisCommand.INFO);
3. RedisBaseClient是通信层。 如果扩展其他用途继承即可,比如RedisPubSub:RedisBaseClient,RedisSentinel:RedisBaseClient
4. 供上层良好调用的话,可以做成partial类扩展redisclient。比如RedisClient.String
5. 订阅的监听使用while的,可在触发事件里面做阻塞。
6. PoolRedisClient池的实现使用ConcurrentStack,仅达到了复用socket连接的目的。
7. 支持socket重连,做法是关闭旧连接,重新建立新socket。
8. 多个命令使用管道实现。见set实现
后续思路:
一:PoolRedisClient池里面连接的释放问题?
1. 使用using
2. 不用使用using,会自动检测并回收 。
不做成自动检测的话,就会出现连接无法释放的问题,总会有人忘记释放的,所以要优化成1+2结合的方式。
二:client池和socket池分离,socket单独做一个池? 还在考虑中。
Hredis设计图
命令执行流程图、解决方案图、类图。

单元测试场景
一. Info命令通信、密码配置。
[TestMethod, TestCategory("Server")]
public void Redis_PassWord()
{
using (var rcClient = new RedisClient(new RedisConfiguration()
{
Host = ip,
Port = ,
PassWord = ""
}))
{
var info = rcClient.Send(RedisCommand.INFO);
Debug.Write(info.ToString());
}
}
二. 普通订阅,及模式匹配订阅。这里订阅的是Sentinel事件信息。
[TestMethod, TestCategory("PushSub")]
public void Subscribe_Sentinel_Test()
{
using (RedisPubSub rsc = new RedisPubSub("127.0.0.1", 20001))
{
rsc.SubscriptionReceived += rsc_SubscriptionReceived;
//rsc.Subscribe("+sdown");
}
}
[TestMethod, TestCategory("PushSub")]
public void PSubscribe_Sentinel_Test()
{
using (RedisPubSub rsc = new RedisPubSub("127.0.0.1", ))
{
rsc.SubscriptionReceived += rsc_SubscriptionReceived;
// rsc.PSubscribe("*");
}
}
private void rsc_SubscriptionReceived(object sender, object args)
{
if (args is object[])
{
var list = args as object[];
foreach (var o in list)
{
Debug.Write("\r\n" + o.ToString());
}
}
else
{
Debug.Write("\r\n" + args.ToString());
}
var sr = sender as RedisPubSub;
sr.UnSubscribe("*");
}
三. client连接池、Parallel并发模拟。
[TestMethod, TestCategory("poolRedisclient")]
public void GetClient_Test()
{
PoolRedisClient prc = new PoolRedisClient(new PoolConfiguration()
{
Host = ip,
Port = port,
MaxClients =
});
using (var client = prc.GetClient())
{
client.Set("GetClient_Test", "GetClient_Test");
var info2 = client.Get("GetClient_Test");
Assert.AreEqual(info2.ToString(), "GetClient_Test");
}
prc.Dispose();
}
[TestMethod, TestCategory("poolRedisclient")]
public void Parallel_PoolClient_Test()
{
PoolRedisClient prc = new PoolRedisClient(new PoolConfiguration()
{
Host = ip,
Port = port,
MaxClients =
});
Parallel.For(, , new ParallelOptions() {MaxDegreeOfParallelism = }, (index, item) =>
{
using (var client = prc.GetClient())
{
Thread.Sleep();
client.Set("Parallel_PoolClient_Test" + index, "Parallel_PoolClient_Test");
var info2 = client.Get("Parallel_PoolClient_Test" + index);
Assert.AreEqual(info2.ToString(), "Parallel_PoolClient_Test");
}
});
prc.Dispose();
}
三. 超时重连、多线程并发超时重连。
[TestMethod, TestCategory("poolRedisclient")]
public void PoolClient_TimeOut_Test()
{
PoolRedisClient prc = new PoolRedisClient(new PoolConfiguration()
{
Host = ip,
Port = port,
MaxClients =
});
object info2;
using (var client = prc.GetClient())
{
var result = client.Set("PoolClient_TimeOut_Test", "PoolClient_TimeOut_Test");
Thread.Sleep();
info2 = client.Get("PoolClient_TimeOut_Test");
}
Assert.AreEqual(info2.ToString(), "PoolClient_TimeOut_Test");
prc.Dispose();
}
[TestMethod, TestCategory("poolRedisclient")]
public void Thread_PoolClient_Test()
{
PoolRedisClient prc = new PoolRedisClient(new PoolConfiguration()
{
Host = ip,
Port = port
});
Parallel.For(, , new ParallelOptions() {MaxDegreeOfParallelism = }, (index, item) =>
{
var t = new Thread(() =>
{
Thread.Sleep();
object info2;
using (var client = prc.GetClient())
{
client.Set("Parallel_PoolClient_Test" + index, "Parallel_PoolClient_Test");
Thread.Sleep();
info2 = client.Get("Parallel_PoolClient_Test" + index);
}
Assert.AreEqual(info2.ToString(), "Parallel_PoolClient_Test");
});
t.Start();
});
Thread.Sleep();
prc.Dispose();
}
四. String类型添加、过期时间添加。
[TestMethod, TestCategory("String")]
public void Set_Get_key()
{
using (var rcClient = new RedisClient(ip, port))
{
rcClient.Set("Set_Get_key", "Set_Get_key");
var info2 = rcClient.Get("Set_Get_key");
Assert.AreEqual(info2.ToString(), "Set_Get_key");
}
}
[TestMethod, TestCategory("String")]
public void Set_key_Expire()
{
using (var rcClient = new RedisClient(ip, port))
{
rcClient.Set("Set_key_Expire", "Set_key_Expire", );
var info1 = rcClient.Get("Set_key_Expire");
Assert.AreEqual(info1.ToString(), "Set_key_Expire");
Thread.Sleep();
var info2 = rcClient.Get("Set_key_Expire");
Assert.AreEqual(info2, null);
}
}
总结
开源地址:https://github.com/mushroomsir/HRedis 有需要的同学,可以参考下。
Hredis后续会跟实际需求来写,如果有更好的实现思路,欢迎一起交流。
Redis系列(二)-Hredis客户端设计及开源的更多相关文章
- Redis系列(二):Redis的数据类型及命令操作
原文链接(转载请注明出处):Redis系列(二):Redis的数据类型及命令操作 Redis 中常用命令 Redis 官方的文档是英文版的,当然网上也有大量的中文翻译版,例如:Redis 命令参考.这 ...
- Redis系列(二):Redis的5种数据结构及其常用命令
上一篇博客,我们讲解了什么是Redis以及在Windows和Linux环境下安装Redis的方法, 没看过的同学可以点击以下链接查看: Redis系列(一):Redis简介及环境安装. 本篇博客我们来 ...
- Redis系列(二)--分布式锁、分布式ID简单实现及思路
分布式锁: Redis可以实现分布式锁,只是讨论Redis的实现思路,而真的实现分布式锁,Zookeeper更加可靠 为什么使用分布式锁: 单机环境下只存在多线程,通过同步操作就可以实现对并发环境的安 ...
- Redis系列二之事务及消息通知
一.事务 Redis中的事务是一组命令的集合.一个事务中的命令要么都执行,要么都不执行. 1.事务简介 事务的原理是先将一个事务的命令发送给Redis,然后再让Redis依次执行这些命令.下面看一个示 ...
- Redis系列二:reids介绍
一.什么是redis.redis有哪些特性.redis有哪些应用场景.redis的版本 1. 什么是redis redis是一种基于键值对(key-value)数据库,其中value可以为string ...
- redis系列二: linux下安装redis
下面介绍在Linux环境下,Redis的安装与配置 一. 安装 1.首先上官网下载Redis 压缩包,地址:http://redis.io/download 下载稳定版3.0即可. 2.通过远程管理工 ...
- Redis系列二 - 数据结构
前言 redis作为我们开发的一大神器,我们接触肯定不会少,但是很多同学也许只会存储String类型的值,这是非常不合理的.在这里,将带大家认识Redis的5中数据结构. 1.问:Redis有那些数据 ...
- Redis系列二(yum切换为网易163)
这个可能和Redis没有直接的关系... 是我在yum install的时候发现centos的yum实在是太慢,上网查了下.网易163有个yum镜像,为了让CentOS6使用速度更快的YUM更新源,可 ...
- Redis系列二 Redis数据库介绍
1.SELECT命令 通过查看配置文件可以知道Redis默认有17个库,从0-16. 默认是在0号库.选择库使用SELECT <dbid>命令.例如选择0号库 SELECT 0 2.DB ...
随机推荐
- 体验Visual Studio 2015 之 MVC - 视图组建
VS2015 PERVIEW中可以创建MVC 项目. 我们可以 发现有几大亮点. 首先我们看目录结构: 当前项目包含两个主要的文件夹:Solution Items .src 很明显src文件夹下为当前 ...
- 转:MYSQL连接字符串参数解析(解释)
被迫转到MySQL数据库,发现读取数据库时,tinyint类型的值都被转化为boolean了,这样大于1的值都丢失,变成true了.查阅资料MySQL中无Boolean类型,都是存储为tinyint了 ...
- 关于Telerik RadGridView 数据列拖动后异常的一种情况
目的: 想实现带有复杂表头(ColumnHeader)的列的动态加载,写了一个用户控件. 问题: 动态加载成功了,显示正常,滚动条也正常,但是一旦进行列的拖动操作之后,程序就挂掉了. 解决尝试: 反复 ...
- MongoDB CURD 介绍
MongoDB是用JSON格式的field和value成对的documents存储数据,documents类似于编程语言中的key value 键值对(例如:dictionaries,hashes,m ...
- d8fs9f
你好 - Helloworld 1. a 2. b 3. c 来自为知笔记(Wiz)
- IE11 上的3个bug
1.IE 11在popstate上无法正常使用,所以,需要使用老方法hashchange.有一个叫History.js的library,是可以解决这个问题.但如果url在"#"后跟 ...
- 扩展BindingList,防止增加、删除项时自动更新界面而不出现“跨线程操作界面控件 corss thread operation”异常
在做界面程序时,常常需要一些数据类,界面元素通过绑定等方式显示出数据,然而由于UI线程不是线程安全的,一般都需要通过Invoke等方式来调用界面控件.但对于数据绑定bindingList而言,没法响应 ...
- 使用Jmeter录制web脚本
1:启动Jmeter: 2:选择测试计划,右键->添加->Threads(users)>线程组 3:选择这个线程组,右键->添加->配置元件->HTTP请求默认值, ...
- python实现最简单的计算器功能源码
import re def calc(formula): formula = re.sub(' ', '', formula) formula_ret = 0 match_brackets = re. ...
- WIN32 窗口类封装 框架实现部分
上面已经讲了窗口封装部分,内容可点击:http://www.cnblogs.com/mengdejun/p/4010320.html,下面分享框架部分内容,完成WINDOWS消息迭代 CQFrameW ...
