自用RedisRepository分享和纠错,本文版权归博客园和作者吴双本人共同所有,转载和爬虫必须注明原文地址:www.cnblogs.com/tdws 。

一.   写在前面

毕业工作后,今天终于能回家了,回想了一些这半年来所做的内容,总是觉得还停留在那么基础的水平 ,在解决各种问题的过程中,自己的创新思路比较少,靠搜索来的比较多 。不想做16年的总结了 ,希望17年能学更多的我爱的技术,看更多的开源代码,能学到更多的设计思想和代码思路,能再更新这两年来对代码的理解。

这篇分享,主要是弥补我之前RedisRepository的不足。如果对Redis基础有疑问,可移步http://www.cnblogs.com/tdws/tag/NoSql/   .

半年前由于我StackExchange.Redis文档阅读不足,所分享的RedisRepository有所错误。下面列举我的主要错误:

错误1,没有单例化ConnectionMultiplexer Redis连接对象,并且我天真的以为给单例对象加锁,在并发情况下,会限制了Redis的性能。

错误2,在主从情况下,我以为在发生手动切换的时候,我们要订阅切换事件,并在事件发生后,动态改变连接对象指向的Endpoint。

当我再一次仔细阅读文档时,才明白我的错误,这是一篇迟到的修正,但是我自用的repository自我感觉还是有很多不足之处,所以我真的需要老司机的指点和建议。

修正1,Redis连接对象创建的代价很大,并且单例加锁并不会影响Redis性能,因为在发生网络请求的期间,连接对象并没有在等待中。

修正2,Redis主从时,在哨兵切换主从关系后,StackExchange.Redis会为我们识别新的主从,不需要我们做任何操作。

目前为止我还有两个疑问。

疑问1,在看文档后没有明确结果。当做主从读写分离时,  我们在Endpoint Collection集合中添加多个节点就会自动读写分离?还是说需要 我们在读取命令的方法中指定CommandFlags.PreferSlave?  我认为是后者吧?所以我在我所有的读取方法都指定了PreferSlave。    老司机们怎么说?

疑问2,我使用LuaScript.Prepare(lua)后再Load出来,执行lua总是无效果,并且LuaScript.GetCachedScriptCount()为0. 不过我直接使用ScriptEvaluateAsync却是好用的,老司机如果有好的例子,希望老司机给些指导或者分享。

二.   概念上有必要加强一下

StackExchange.Redis客户端所提供的功能,绝不是使用Socket连接下,用socket对象send点数据,recv点数据这么简单。也不是说创建包含几个redis socket连接的连接池那么容易,StackExchange的目标在于管道处理和多路复用。关于连接数量的多少,可能是一个也可能是一个连接池,在于我们如何使用,可以参考老外这篇文章https://frankdecaire.blogspot.jp/2016/03/stack-exchange-redis-connection-pooling.html  ,相反的如果没有管道和多路复用的思想,由于socket面向有连接的协议时,send和recv都是阻塞的,他们在System.Net.Socket实现的内部调用了win32 API WSARecv和WSASend。连续的200个send recv对,最后一个命令就要在前面199个send&recv执行结束后发送到redis。所以有了StackExchange.Redis 我们不必把所有的时间都浪费在客户端到服务端以及服务端到客户端的等待上。当不同调用方同一时刻访问时,StackExchange客户端会自动使用管道分离访问请求,因此无论使用阻塞还是异步的访问方式,这些工作都是被管道处理的。从本质上讲,它填补了 waiting 时间与其他调用方的工作。

三.   代码结构,仅供参考

结构大概就是这样,RedisAsyncHelper下的所有类都是部分类,他们的类名称是RedisHelper。他们共同实现了IRedisHelper的接口,并且留下了详细的注释。

同步版本和异步版本的目录结构是一样的。所以我仅分享下Async版本,如果对Async有疑问可移步 http://www.cnblogs.com/tdws/p/6172207.html

四.   预备阶段

CommonHelper中的两个帮助类:

RedisInnerTypeHelper.cs

  1. using StackExchange.Redis;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4.  
  5. namespace Fantasy.RedisRepository.CommonHelper
  6. {
  7. internal class RedisInnerTypeHelper
  8. {
  9. public static List<T> RedisValuesToGenericList<T>(RedisValue[] redisValues)
  10. {
  11. var result = new List<T>();
  12. redisValues.ToList().ForEach(r => result.Add(SerializeHelper.Deserialize<T>(r)));
  13. return result;
  14. }
  15.  
  16. public static RedisValue[] GenericListToRedisValues<T>(List<T> values)
  17. {
  18. var redisValues = new List<RedisValue>();
  19. values.ForEach(v => redisValues.Add(SerializeHelper.Serialize(values)));
  20. return redisValues.ToArray();
  21. }
  22.  
  23. public static RedisKey[] GenericListToRedisKeys(List<string> keys)
  24. {
  25. var redisKeys = new List<RedisKey>();
  26. keys.ForEach(k => redisKeys.Add(k));
  27. return redisKeys.ToArray();
  28. }
  29. }
  30. }

SerializeHelper.cs

  1. using System.IO;
  2. using System.Runtime.Serialization.Formatters.Binary;
  3.  
  4. namespace Fantasy.RedisRepository.CommonHelper
  5. {
  6. internal static class SerializeHelper
  7. {
  8. /// <summary>
  9. /// 字节数组序列化
  10. /// </summary>
  11. /// <param name="o"></param>
  12. /// <returns></returns>
  13. internal static byte[] Serialize(object o)
  14. {
  15. if (o == null)
  16. {
  17. return null;
  18. }
  19.  
  20. BinaryFormatter binaryFormatter = new BinaryFormatter();
  21. using (MemoryStream memoryStream = new MemoryStream())
  22. {
  23. binaryFormatter.Serialize(memoryStream, o);
  24. byte[] objectDataAsStream = memoryStream.ToArray();
  25. return objectDataAsStream;
  26. }
  27. }
  28.  
  29. /// <summary>
  30. /// 字节数组反序列化
  31. /// </summary>
  32. /// <typeparam name="T"></typeparam>
  33. /// <param name="stream"></param>
  34. /// <returns></returns>
  35. internal static T Deserialize<T>(byte[] stream)
  36. {
  37. if (stream == null)
  38. {
  39. return default(T);
  40. }
  41.  
  42. BinaryFormatter binaryFormatter = new BinaryFormatter();
  43. using (MemoryStream memoryStream = new MemoryStream(stream))
  44. {
  45. T result = (T)binaryFormatter.Deserialize(memoryStream);
  46. return result;
  47. }
  48. }
  49. }
  50. }

Config中的配置类:

ConfigHelper.cs

  1. using System;
  2. using System.Configuration;
  3.  
  4. namespace Fantasy.RedisRepository.Config
  5. {
  6. internal class ConfigHelper
  7. {
  8. internal static T Get<T>(string appSettingsKey, T defaultValue)
  9. {
  10. string text = ConfigurationManager.AppSettings[appSettingsKey];
  11. if (string.IsNullOrWhiteSpace(text))
  12. return defaultValue;
  13. try
  14. {
  15. var value = Convert.ChangeType(text, typeof(T));
  16. return (T)value;
  17. }
  18. catch
  19. {
  20. return defaultValue;
  21. }
  22. }
  23. }
  24. }

RedisClientConfig.cs

  1. namespace Fantasy.RedisRepository.Config
  2. {
  3. internal class RedisClientConfig
  4. {
  5.  
  6. private static string _server = ConfigHelper.Get("RedisServer", "115.xx.xx.31");
  7. /// <summary>
  8. /// 节点IP
  9. /// </summary>
  10. public static string Server
  11. {
  12. get { return _server; }
  13. set { _server = value; }
  14. }
  15.  
  16. private static int _port = ConfigHelper.Get("RedisPort", );
  17. /// <summary>
  18. /// 节点端口
  19. /// </summary>
  20. public static int Port
  21. {
  22. get { return _port; }
  23. set { _port = value; }
  24. }
  25.  
  26. private static string _slaveServer = ConfigHelper.Get("SlaveServer", "115.xx.xx.31");
  27. /// <summary>
  28. /// 节点IP
  29. /// </summary>
  30. public static string SlaveServer
  31. {
  32. get { return _slaveServer; }
  33. set { _slaveServer = value; }
  34. }
  35.  
  36. private static int _slavePort = ConfigHelper.Get("SlavePort", );
  37. /// <summary>
  38. /// 节点端口
  39. /// </summary>
  40. public static int SlavePort
  41. {
  42. get { return _slavePort; }
  43. set { _slavePort = value; }
  44. }
  45.  
  46. private static string _auth = ConfigHelper.Get("RedisAuth", "fantasy..");
  47. /// <summary>
  48. /// 节点密码
  49. /// </summary>
  50. public static string RedisAuth
  51. {
  52. get { return _auth; }
  53. set { _auth = value; }
  54. }
  55.  
  56. private static int _defaultDatabase = ConfigHelper.Get("RedisDataBase", );
  57. /// <summary>
  58. /// redis默认0号库
  59. /// </summary>
  60. public static int DefaultDatabase
  61. {
  62. get { return _defaultDatabase; }
  63. set { _defaultDatabase = value; }
  64. }
  65.  
  66. private static int _connectTimeout = ;
  67. public static int ConnectTimeout
  68. {
  69. get { return _connectTimeout; }
  70. set { _connectTimeout = value; }
  71. }
  72.  
  73. private static int _connectRetry = ;
  74. public static int ConnectRetry
  75. {
  76. get { return _connectRetry; }
  77. set { _connectRetry = value; }
  78. }
  79.  
  80. private static bool _preserveAsyncOrder = false;
  81. public static bool PreserveAsyncOrder
  82. {
  83. get { return _preserveAsyncOrder; }
  84. set { _preserveAsyncOrder = value; }
  85. }
  86. }
  87. }

RedisConnection.cs

  1. using Fantasy.RedisRepository.Config;
  2. using StackExchange.Redis;
  3.  
  4. namespace Fantasy.RedisRepository
  5. {
  6. /// <summary>
  7. /// Redis连接类
  8. /// </summary>
  9. public static class RedisConnection
  10. {
  11. private static ConnectionMultiplexer _connection;
  12. private static readonly object SyncObject = new object();
  13. /// <summary>
  14. /// redis连接对象,单例加锁不影响性能
  15. /// </summary>
  16. public static ConnectionMultiplexer GenerateConnection
  17. {
  18. get
  19. {
  20. if (_connection == null || !_connection.IsConnected)
  21. {
  22. lock (SyncObject)
  23. {
  24. var configurationOptions = new ConfigurationOptions()
  25. {
  26. Password = RedisClientConfig.RedisAuth,
  27. EndPoints =
  28. {
  29. {RedisClientConfig.Server, RedisClientConfig.Port},
  30. {RedisClientConfig.SlaveServer, RedisClientConfig.SlavePort}
  31. }
  32. };
  33. _connection = ConnectionMultiplexer.Connect(configurationOptions);
  34. }
  35. }
  36. return _connection;
  37. }
  38. }
  39. }
  40. }

五.   RedisHelper

实际上就是做了层序列化包装而已。

IRedisHelper:

  1. using System;
  2. using StackExchange.Redis;
  3. using System.Collections.Generic;
  4. using System.Threading.Tasks;
  5.  
  6. namespace Fantasy.RedisRepository.RedisHelpers
  7. {
  8. /// <summary>
  9. /// 异步方法接口 --Author 吴双 www.cnblogs.com/tdws
  10. /// 存入数据均为方法内部序列化后的byte,所以取数据的时候需要反序列化时,请指定正确的数据类型
  11. /// </summary>
  12. public partial interface IRedisHelper
  13. {
  14. #region Redis数据类型—String
  15.  
  16. /// <summary>
  17. /// 将任何数据以redis string存储
  18. /// </summary>
  19. /// <typeparam name="T"></typeparam>
  20. /// <param name="key"></param>
  21. /// <param name="value"></param>
  22. /// <param name="timeout"></param>
  23. /// <returns></returns>
  24. Task<bool> StringSetAsync<T>(string key, T value, TimeSpan? timeout = null);
  25.  
  26. /// <summary>
  27. /// 对数值进行减法操作,默认-1
  28. /// </summary>
  29. /// <param name="key"></param>
  30. /// <param name="value"></param>
  31. /// <returns>操作后的结果</returns>
  32. Task<long> StringDecrementAsync(string key, long value = 1L);
  33.  
  34. /// <summary>
  35. /// 对数值进行加法操作,默认+1
  36. /// </summary>
  37. /// <param name="key"></param>
  38. /// <param name="value"></param>
  39. /// <returns>操作后的结果</returns>
  40. Task<long> StringIncrementAsync(string key, long value = 1L);
  41.  
  42. /// <summary>
  43. /// 从redis string中以指定类型取出
  44. /// </summary>
  45. /// <typeparam name="T"></typeparam>
  46. /// <param name="key"></param>
  47. /// <returns></returns>
  48. Task<T> StringGetAsync<T>(string key);
  49.  
  50. #endregion
  51.  
  52. #region Redis数据类型—Hash
  53.  
  54. /// <summary>
  55. /// 向Hash key中存储任意类型任意值
  56. /// </summary>
  57. /// <typeparam name="T"></typeparam>
  58. /// <param name="key"></param>
  59. /// <param name="field"></param>
  60. /// <param name="value"></param>
  61. /// <returns>是否成功</returns>
  62. Task<bool> HashSetAsync<T>(string key, string field, T value);
  63.  
  64. /// <summary>
  65. /// 批量 向Hash key中存储任意类型任意值
  66. /// </summary>
  67. /// <typeparam name="T"></typeparam>
  68. /// <param name="key"></param>
  69. /// <param name="hashFields"></param>
  70. /// <returns>无返回值</returns>
  71. Task HashMultiSetAsync<T>(string key, Dictionary<string, T> hashFields);
  72.  
  73. /// <summary>
  74. /// 对指定hash key中制定field做数量增加操作 默认自增1
  75. /// 如果此操作前key不存在 则创建。 如果此操作前该field不存在或者非数字 则先被置0,再被继续操作
  76. /// </summary>
  77. /// <param name="key"></param>
  78. /// <param name="field"></param>
  79. /// <param name="incrCount"></param>
  80. /// <returns>操作后的结果</returns>
  81. Task<long> HashIncrementAsync(string key, string field, long incrCount = );
  82.  
  83. /// <summary>
  84. /// 对指定hash key中制定field做数量增加操作 默认自减1
  85. /// 如果此操作前key不存在 则创建。 如果此操作前该field不存在或者非数字 则先被置0,再被继续操作
  86. /// </summary>
  87. /// <param name="key"></param>
  88. /// <param name="field"></param>
  89. /// <param name="decrCount"></param>
  90. /// <returns>操作后的结果</returns>
  91. Task<long> HashDecrementAsync(string key, string field, long decrCount = );
  92.  
  93. /// <summary>
  94. /// 从指定Hash中 删除指定field
  95. /// 如果key或者field不存在,则false
  96. /// </summary>
  97. /// <param name="key"></param>
  98. /// <param name="field"></param>
  99. /// <returns>是否成功</returns>
  100. Task<bool> HashDeleteFieldAsync(string key, string field);
  101.  
  102. /// <summary>
  103. /// 从指定Hash key中 批量删除指定field
  104. /// 如果key或者field不存在,则false
  105. /// </summary>
  106. /// <param name="key"></param>
  107. /// <param name="fields"></param>
  108. /// <returns>移除数量</returns>
  109. Task<long> HashMultiDeleteFieldAsync(string key, List<string> fields);
  110.  
  111. /// <summary>
  112. /// 从指定Hash key中获取指定field值
  113. /// </summary>
  114. /// <typeparam name="T"></typeparam>
  115. /// <param name="key"></param>
  116. /// <param name="field"></param>
  117. /// <returns></returns>
  118. Task<T> HashGetAsync<T>(string key, string field);
  119.  
  120. /// <summary>
  121. /// 从指定Hash key中判断field是否存在
  122. /// </summary>
  123. /// <param name="key"></param>
  124. /// <param name="field"></param>
  125. /// <returns></returns>
  126. Task<bool> HashFieldExistAsync(string key, string field);
  127.  
  128. /// <summary>
  129. /// 获取指定Hash key中的所有field的值
  130. /// </summary>
  131. /// <typeparam name="T"></typeparam>
  132. /// <param name="key"></param>
  133. /// <returns></returns>
  134. Task<List<T>> HashValuesAsync<T>(string key);
  135.  
  136. /// <summary>
  137. /// 获取指定Hash key中所有 field名称及其Value
  138. /// </summary>
  139. /// <typeparam name="T"></typeparam>
  140. /// <param name="key"></param>
  141. /// <returns></returns>
  142. Task<Dictionary<string, T>> HashGetAllAsync<T>(string key);
  143.  
  144. /// <summary>
  145. /// 获取指定Hash key中所有field
  146. /// </summary>
  147. /// <param name="key"></param>
  148. /// <returns></returns>
  149. Task<List<string>> HashFieldsAsync(string key);
  150.  
  151. #endregion
  152.  
  153. #region Redis数据类型—List
  154.  
  155. /// <summary>
  156. /// 在指定pivot后插入value, 如果pivot不存在,则返回-1, 如果key不存在,则返回0
  157. /// 如果存在多个相同指定的的pivot,则插入第一个指定pivot后面.
  158. /// 即链表从左向右查找,遇到指定pivot,则确定位置
  159. /// </summary>
  160. /// <typeparam name="T"></typeparam>
  161. /// <param name="key"></param>
  162. /// <param name="pivot">list中的一个值</param>
  163. /// <param name="value"></param>
  164. /// <returns></returns>
  165. Task<long> ListInsertAfterAsync<T>(string key, string pivot, T value);
  166.  
  167. /// <summary>
  168. /// 在指定pivot前插入value, 如果pivot不存在,则返回-1, 如果key不存在,则返回0
  169. /// 如果存在多个相同指定的的pivot,则插入第一个指定pivot前面.
  170. /// 即链表从左向右查找,遇到指定pivot,则确定位置
  171. /// </summary>
  172. /// <typeparam name="T"></typeparam>
  173. /// <param name="key"></param>
  174. /// <param name="pivot"></param>
  175. /// <param name="value"></param>
  176. /// <returns></returns>
  177. Task<long> ListInsertBeforeAsync<T>(string key, string pivot, T value);
  178.  
  179. /// <summary>
  180. /// 从链表左侧弹出第一个元素(弹出能获取到该元素并且被删除)
  181. /// 如果key不存在 或者链表为空 则为null
  182. /// </summary>
  183. /// <typeparam name="T"></typeparam>
  184. /// <param name="key"></param>
  185. /// <returns></returns>
  186. Task<T> ListLeftPopAsync<T>(string key);
  187.  
  188. /// <summary>
  189. /// 从链表左侧增加一个元素,key不存在则被创建
  190. /// </summary>
  191. /// <typeparam name="T"></typeparam>
  192. /// <param name="key"></param>
  193. /// <param name="value"></param>
  194. /// <returns>返回操作后的链表长度</returns>
  195. Task<long> ListLeftPushAsync<T>(string key, T value);
  196.  
  197. /// <summary>
  198. /// 从链表左侧批量增加元素,如果 a b c 则c会在链表左侧第一位 b第二位 a第三位
  199. /// </summary>
  200. /// <typeparam name="T"></typeparam>
  201. /// <param name="key"></param>
  202. /// <param name="values"></param>
  203. /// <returns>返回操作后的链表长度</returns>
  204. Task<long> ListLeftMultiPushAsync<T>(string key, List<T> values);
  205.  
  206. /// <summary>
  207. /// 获取链表长度,不存在key则为0
  208. /// </summary>
  209. /// <typeparam name="T"></typeparam>
  210. /// <param name="key"></param>
  211. /// <returns></returns>
  212. Task<long> ListLengthAsync<T>(string key);
  213.  
  214. /// <summary>
  215. /// 获取链表中所有数据,从左侧start开始到stop结束,从0—-1则认为获取全部,默认获取全部
  216. /// start为负数则代表从链表右侧开始,-1为右侧第一位,-2为右侧第二位
  217. /// start要小于stop,否则返回null
  218. /// </summary>
  219. /// <typeparam name="T"></typeparam>
  220. /// <param name="key"></param>
  221. /// <param name="start"></param>
  222. /// <param name="stop"></param>
  223. /// <returns></returns>
  224. Task<List<T>> ListRangeAsync<T>(string key, long start = 0L, long stop = -1L);
  225.  
  226. /// <summary>
  227. /// 从链表中一处count数量的value. count大于0则从左至右,count小于0则从右至左,count=0则移除全部
  228. /// </summary>
  229. /// <typeparam name="T"></typeparam>
  230. /// <param name="key"></param>
  231. /// <param name="value"></param>
  232. /// <param name="count"></param>
  233. /// <returns></returns>
  234. Task<long> ListRemoveAsync<T>(string key, T value, long count = 0L);
  235.  
  236. /// <summary>
  237. /// 从右侧弹出第一个元素(弹出能获取到该元素并且被删除)
  238. /// </summary>
  239. /// <typeparam name="T"></typeparam>
  240. /// <param name="key"></param>
  241. /// <returns></returns>
  242. Task<T> ListRightPopAsync<T>(string key);
  243.  
  244. /// <summary>
  245. /// 从链表右侧加入元素,如果 rpush a b c 则c为右侧第一位 b第二位 c第三位
  246. /// </summary>
  247. /// <typeparam name="T"></typeparam>
  248. /// <param name="key"></param>
  249. /// <param name="value"></param>
  250. /// <returns></returns>
  251. Task<long> ListRightPushAsync<T>(string key, T value);
  252.  
  253. /// <summary>
  254. /// 从右侧批量插入,和左侧相反
  255. /// </summary>
  256. /// <typeparam name="T"></typeparam>
  257. /// <param name="key"></param>
  258. /// <param name="values"></param>
  259. /// <returns></returns>
  260. Task<long> ListRightMultiPushAsync<T>(string key, List<T> values);
  261.  
  262. /// <summary>
  263. /// 在链表指定索引处,插入元素
  264. /// 正数索引从0开始,代表左侧。负数从-1开始 代表从右侧。-1为右侧第一位
  265. /// </summary>
  266. /// <typeparam name="T"></typeparam>
  267. /// <param name="key"></param>
  268. /// <param name="index"></param>
  269. /// <param name="value"></param>
  270. /// <returns></returns>
  271. Task ListSetByIndexAsync<T>(string key, int index, T value);
  272.  
  273. /// <summary>
  274. /// 留下start到stop之间的数据。负数代表从右侧寻找 -1为右侧第一位
  275. /// </summary>
  276. /// <param name="key"></param>
  277. /// <param name="start"></param>
  278. /// <param name="stop"></param>
  279. /// <returns></returns>
  280. Task ListTrimAsync(string key, long start, long stop);
  281.  
  282. /// <summary>
  283. /// 获取指定index的值,负数代表从右侧寻找 -1为右侧第一位
  284. /// </summary>
  285. /// <typeparam name="T"></typeparam>
  286. /// <param name="key"></param>
  287. /// <param name="index"></param>
  288. /// <returns></returns>
  289. Task<T> ListGetByIndexAsync<T>(string key, long index);
  290.  
  291. #endregion
  292.  
  293. #region Redis数据类型—Set
  294.  
  295. /// <summary>
  296. /// 向指定集合中增加一个元素
  297. /// </summary>
  298. /// <typeparam name="T"></typeparam>
  299. /// <param name="key"></param>
  300. /// <param name="value"></param>
  301. /// <returns></returns>
  302. Task<bool> SetAddAsync<T>(string key, T value);
  303.  
  304. /// <summary>
  305. /// 指定集合计算操作operation枚举,指定计算结果将存的目标destKey,指定需要参与计算的多个key
  306. /// </summary>
  307. /// <param name="operation"></param>
  308. /// <param name="destKey"></param>
  309. /// <param name="combineKeys"></param>
  310. /// <returns></returns>
  311. Task<long> SetCombineAndStoreAsync(SetOperation operation, string destKey, List<string> combineKeys);
  312.  
  313. /// <summary>
  314. /// 指定集合计算操作operation枚举,指定需要参与计算的多个key
  315. /// </summary>
  316. /// <typeparam name="T"></typeparam>
  317. /// <param name="operation"></param>
  318. /// <param name="combineKeys"></param>
  319. /// <returns></returns>
  320. Task<List<T>> SetCombineAsync<T>(SetOperation operation, List<string> combineKeys);
  321.  
  322. /// <summary>
  323. /// 指定值是否存在于指定集合中
  324. /// </summary>
  325. /// <typeparam name="T"></typeparam>
  326. /// <param name="key"></param>
  327. /// <param name="value"></param>
  328. /// <returns></returns>
  329. Task<bool> SetContainsAsync<T>(string key, T value);
  330.  
  331. /// <summary>
  332. /// 获取指定集合中元素个数
  333. /// </summary>
  334. /// <param name="key"></param>
  335. /// <returns></returns>
  336. Task<long> SetLengthAsync(string key);
  337.  
  338. /// <summary>
  339. /// 获取指定集合中的所有元素
  340. /// </summary>
  341. /// <typeparam name="T"></typeparam>
  342. /// <param name="key"></param>
  343. /// <param name="value"></param>
  344. /// <returns></returns>
  345. Task<List<T>> SetMembersAsync<T>(string key, T value);
  346.  
  347. /// <summary>
  348. /// 从sourceKey移除指定value到目标distKey集合当中
  349. /// 如果sourceKey存在指定value则返回true,否则不做任何操作返回false
  350. /// </summary>
  351. /// <typeparam name="T"></typeparam>
  352. /// <param name="sourcekey"></param>
  353. /// <param name="distKey"></param>
  354. /// <param name="value"></param>
  355. /// <returns></returns>
  356. Task<bool> SetMoveAsync<T>(string sourcekey, string distKey, T value);
  357.  
  358. /// <summary>
  359. /// 从指定集合当中随机取出一个元素
  360. /// </summary>
  361. /// <typeparam name="T"></typeparam>
  362. /// <param name="key"></param>
  363. /// <returns></returns>
  364. Task<T> SetRandomMemberAsync<T>(string key);
  365.  
  366. /// <summary>
  367. /// 从指定集合随机弹出(删除并获取)一个元素
  368. /// </summary>
  369. /// <typeparam name="T"></typeparam>
  370. /// <param name="key"></param>
  371. /// <returns></returns>
  372. Task<T> SetPopAsync<T>(string key);
  373.  
  374. /// <summary>
  375. /// 从集合中随机弹出(删除并获取)多个元素
  376. /// </summary>
  377. /// <typeparam name="T"></typeparam>
  378. /// <param name="key"></param>
  379. /// <returns></returns>
  380. Task<List<T>> SetRandomMembersAsync<T>(string key);
  381.  
  382. /// <summary>
  383. /// 从集合中移除指定元素
  384. /// </summary>
  385. /// <typeparam name="T"></typeparam>
  386. /// <param name="key"></param>
  387. /// <param name="value"></param>
  388. /// <returns></returns>
  389. Task<bool> SetRemoveAsync<T>(string key, T value);
  390.  
  391. /// <summary>
  392. /// 从集合中批量移除元素
  393. /// </summary>
  394. /// <typeparam name="T"></typeparam>
  395. /// <param name="key"></param>
  396. /// <param name="values"></param>
  397. /// <returns></returns>
  398. Task<long> SetMultiRemoveAsync<T>(string key, List<T> values);
  399.  
  400. #endregion
  401.  
  402. #region Redis数据类型—SortSet
  403.  
  404. #endregion
  405.  
  406. #region Redis Key操作
  407.  
  408. /// <summary>
  409. /// 删除指定key
  410. /// </summary>
  411. /// <param name="key"></param>
  412. /// <returns></returns>
  413. Task<bool> KeyDeleteAsync(string key);
  414.  
  415. /// <summary>
  416. /// 设置key过期时间具体DateTime
  417. /// </summary>
  418. /// <param name="key"></param>
  419. /// <param name="expireAt"></param>
  420. /// <returns></returns>
  421. Task<bool> KeyExpireAtAsync(string key, DateTime expireAt);
  422.  
  423. /// <summary>
  424. /// 设置key在将来的timeout后过期(TimeSpan)
  425. /// </summary>
  426. /// <param name="key"></param>
  427. /// <param name="timeout"></param>
  428. /// <returns></returns>
  429. Task<bool> KeyExpireInAsync(string key, TimeSpan timeout);
  430.  
  431. /// <summary>
  432. /// key重命名
  433. /// </summary>
  434. /// <param name="key"></param>
  435. /// <param name="newKey"></param>
  436. /// <returns></returns>
  437. Task<bool> KeyRenameAsync(string key, string newKey);
  438.  
  439. /// <summary>
  440. /// 判断key是否已存在
  441. /// </summary>
  442. /// <param name="key"></param>
  443. /// <returns></returns>
  444. Task<bool> KeyExistsAsync(string key);
  445.  
  446. #endregion
  447.  
  448. #region Redis Transcation
  449.  
  450. /// <summary>
  451. /// 在事务中执行一系列redis命令。注意:在委托中的一系列命令的所有 值 都需要进行字节数组序列化
  452. /// </summary>
  453. /// <param name="ranOperations"></param>
  454. /// <returns></returns>
  455. Task<bool> DoInTranscationAsync(Action<ITransaction> ranOperations);
  456.  
  457. #endregion
  458.  
  459. Task<RedisResult> Test();
  460. }
  461. }

RedisHelper部分类RedisStringHelperAsync.cs

  1. using System;
  2. using Fantasy.RedisRepository.CommonHelper;
  3. using StackExchange.Redis;
  4. using System.Threading.Tasks;
  5.  
  6. namespace Fantasy.RedisRepository.RedisHelpers
  7. {
  8. /// <summary>
  9. /// Redis异步操作类 String部分类
  10. /// </summary>
  11. internal partial class RedisHelper// : IRedisHelper
  12. {
  13.  
  14. private static IDatabase _client;
  15.  
  16. internal RedisHelper()
  17. {
  18. _client = RedisConnection.GenerateConnection.GetDatabase();
  19. }
  20.  
  21. #region String 写操作
  22.  
  23. /// <summary>
  24. /// 将任何数据添加到redis中
  25. /// </summary>
  26. /// <typeparam name="T"></typeparam>
  27. /// <param name="key"></param>
  28. /// <param name="value"></param>
  29. /// <param name="timeout"></param>
  30. /// <returns></returns>
  31. public async Task<bool> StringSetAsync<T>(string key, T value, TimeSpan? timeout = null)
  32. {
  33. return await _client.StringSetAsync(key, SerializeHelper.Serialize(value), timeout);
  34. }
  35.  
  36. public async Task<long> StringDecrementAsync(string key, long value = 1L)
  37. {
  38. return await _client.StringDecrementAsync(key, value);
  39. }
  40.  
  41. public async Task<long> StringIncrementAsync(string key, long value = 1L)
  42. {
  43. return await _client.StringIncrementAsync(key, value);
  44. }
  45. #endregion
  46.  
  47. #region String 读操作
  48.  
  49. /// <summary>
  50. /// 根据key获取指定类型数据
  51. /// </summary>
  52. /// <typeparam name="T"></typeparam>
  53. /// <param name="key"></param>
  54. /// <returns></returns>
  55. public async Task<T> StringGetAsync<T>(string key)
  56. {
  57. return SerializeHelper.Deserialize<T>(await _client.StringGetAsync(key, CommandFlags.PreferSlave));
  58. }
  59.  
  60. #endregion
  61.  
  62. }
  63. }

RedisHelper部分类RedisHashHelperAsync.cs

  1. using Fantasy.RedisRepository.CommonHelper;
  2. using StackExchange.Redis;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Threading.Tasks;
  6.  
  7. namespace Fantasy.RedisRepository.RedisHelpers
  8. {
  9. /// <summary>
  10. /// Redis异步操作类 Hash部分类
  11. /// </summary>
  12. internal partial class RedisHelper
  13. {
  14. #region Hash 写操作
  15.  
  16. public async Task<bool> HashSetAsync<T>(string key, string field, T value)
  17. {
  18. return await _client.HashSetAsync(key, field, SerializeHelper.Serialize(value));
  19. }
  20.  
  21. public async Task HashMultiSetAsync<T>(string key, Dictionary<string, T> hashFields)
  22. {
  23. List<HashEntry> entries = new List<HashEntry>();
  24. hashFields.ToList().ForEach(d => entries.Add(new HashEntry(d.Key, SerializeHelper.Serialize(d.Value))));
  25. await _client.HashSetAsync(key, entries.ToArray());
  26. }
  27.  
  28. public async Task<long> HashIncrementAsync(string key, string field, long incrCount = )
  29. {
  30. return await _client.HashIncrementAsync(key, field, incrCount);
  31. }
  32.  
  33. public async Task<long> HashDecrementAsync(string key, string field, long decrCount = )
  34. {
  35. return await _client.HashDecrementAsync(key, field, decrCount);
  36. }
  37.  
  38. public async Task<bool> HashDeleteFieldAsync(string key, string field)
  39. {
  40. return await _client.HashDeleteAsync(key, field);
  41. }
  42.  
  43. public async Task<long> HashMultiDeleteFieldAsync(string key, List<string> fields)
  44. {
  45. List<RedisValue> values = new List<RedisValue>();
  46. fields.ForEach(f => values.Add(f));
  47. return await _client.HashDeleteAsync(key, values.ToArray());
  48. }
  49.  
  50. #endregion
  51.  
  52. #region Hash 读操作
  53.  
  54. /// <summary>
  55. /// Redis 指定hash类型key中field是否存在
  56. /// </summary>
  57. /// <param name="key"></param>
  58. /// <param name="field"></param>
  59. /// <returns></returns>
  60. public async Task<bool> HashFieldExistAsync(string key, string field)
  61. {
  62. return await _client.HashExistsAsync(key, field, CommandFlags.PreferSlave);
  63. }
  64. public async Task<List<string>> HashFieldsAsync(string key)
  65. {
  66. RedisValue[] values = await _client.HashKeysAsync(key, CommandFlags.PreferSlave);
  67. return RedisInnerTypeHelper.RedisValuesToGenericList<string>(values);
  68. }
  69. public async Task<List<T>> HashValuesAsync<T>(string key)
  70. {
  71. var values = await _client.HashValuesAsync(key, CommandFlags.PreferSlave);
  72. return RedisInnerTypeHelper.RedisValuesToGenericList<T>(values);
  73. }
  74.  
  75. public async Task<T> HashGetAsync<T>(string key, string field)
  76. {
  77. return SerializeHelper.Deserialize<T>(await _client.HashGetAsync(key, field, CommandFlags.PreferSlave));
  78. }
  79.  
  80. public async Task<Dictionary<string, T>> HashGetAllAsync<T>(string key)
  81. {
  82. HashEntry[] entries = await _client.HashGetAllAsync(key, CommandFlags.PreferSlave);
  83. Dictionary<string, T> dic = new Dictionary<string, T>();
  84. entries.ToList().ForEach(e => dic.Add(e.Name, SerializeHelper.Deserialize<T>(e.Value)));
  85. return dic;
  86. }
  87.  
  88. #endregion
  89.  
  90. }
  91. }

RedisHelper部分类RedisListHelperAsync.cs

  1. using Fantasy.RedisRepository.CommonHelper;
  2. using StackExchange.Redis;
  3. using System.Collections.Generic;
  4. using System.Threading.Tasks;
  5.  
  6. namespace Fantasy.RedisRepository.RedisHelpers
  7. {
  8. /// <summary>
  9. /// Redis异步操作类 List部分类
  10. /// </summary>
  11. internal partial class RedisHelper
  12. {
  13. // _client.ListRightPopLeftPushAsync();
  14. // _client.ListTrimAsync();
  15.  
  16. #region List 写操作
  17.  
  18. public async Task<long> ListInsertAfterAsync<T>(string key, string pivot, T value)
  19. {
  20. return await _client.ListInsertAfterAsync(key, pivot, SerializeHelper.Serialize(value));
  21. }
  22.  
  23. public async Task<long> ListInsertBeforeAsync<T>(string key, string pivot, T value)
  24. {
  25. return await _client.ListInsertBeforeAsync(key, pivot, SerializeHelper.Serialize(value));
  26. }
  27.  
  28. public async Task<T> ListLeftPopAsync<T>(string key)
  29. {
  30. return SerializeHelper.Deserialize<T>(await _client.ListLeftPopAsync(key));
  31. }
  32.  
  33. public async Task<long> ListLeftPushAsync<T>(string key, T value)
  34. {
  35. return await _client.ListLeftPushAsync(key, SerializeHelper.Serialize(value));
  36. }
  37.  
  38. public async Task<long> ListLeftMultiPushAsync<T>(string key, List<T> values)
  39. {
  40. return await _client.ListLeftPushAsync(key, SerializeHelper.Serialize(RedisInnerTypeHelper.GenericListToRedisValues(values)));
  41. }
  42.  
  43. public async Task<long> ListRemoveAsync<T>(string key, T value, long count = 0L)
  44. {
  45. return await _client.ListRemoveAsync(key, SerializeHelper.Serialize(value), count);
  46. }
  47.  
  48. public async Task<T> ListRightPopAsync<T>(string key)
  49. {
  50. return SerializeHelper.Deserialize<T>(await _client.ListRightPopAsync(key));
  51. }
  52.  
  53. public async Task<long> ListRightPushAsync<T>(string key, T value)
  54. {
  55. return await _client.ListRightPushAsync(key, SerializeHelper.Serialize(value));
  56. }
  57.  
  58. public async Task<long> ListRightMultiPushAsync<T>(string key, List<T> values)
  59. {
  60. return
  61. await
  62. _client.ListRightPushAsync(key,
  63. SerializeHelper.Serialize(RedisInnerTypeHelper.GenericListToRedisValues(values)));
  64. }
  65.  
  66. public async Task ListSetByIndexAsync<T>(string key, int index, T value)
  67. {
  68. await _client.ListSetByIndexAsync(key, index, SerializeHelper.Serialize(value));
  69. }
  70.  
  71. public async Task ListTrimAsync(string key, long start, long stop)
  72. {
  73. await _client.ListTrimAsync(key, start, stop);
  74. }
  75. #endregion
  76.  
  77. #region List 读操作
  78.  
  79. public async Task<T> ListGetByIndexAsync<T>(string key, long index)
  80. {
  81. return SerializeHelper.Deserialize<T>(await _client.ListGetByIndexAsync(key, index, CommandFlags.PreferSlave));
  82. }
  83.  
  84. public async Task<long> ListLengthAsync<T>(string key)
  85. {
  86. return await _client.ListLengthAsync(key, CommandFlags.PreferSlave);
  87. }
  88.  
  89. public async Task<List<T>> ListRangeAsync<T>(string key, long start = 0L, long stop = -1L)
  90. {
  91. return RedisInnerTypeHelper.RedisValuesToGenericList<T>(await _client.ListRangeAsync(key, start, stop, CommandFlags.PreferSlave));
  92. }
  93.  
  94. #endregion
  95. }
  96. }

RedisLuaHelper.cs 这里打算装一些功能行lua脚本, 外部依然是传key一类的参数,这个不完整,只是个实例。

  1. using StackExchange.Redis;
  2. using System.Threading.Tasks;
  3.  
  4. namespace Fantasy.RedisRepository.RedisHelpers
  5. {
  6. internal partial class RedisHelper
  7. {
  8. public async Task<RedisResult> LuaMutilGetHash()
  9. {
  10. string lua = @"local result={}
  11. for i, v in ipairs(KEYS) do
  12. result[i] = redis.call('hgetall',v)
  13. end
  14. return result";
  15. var res = await _client.ScriptEvaluateAsync(lua, new RedisKey[] { "people:1", "people:2", "people:3" });
  16. var res1= LuaScript.GetCachedScriptCount();
  17. return res;
  18. }
  19. }
  20. }

关于Transcation的封装,我个人没有什么好的方法,提供了这样一个方法

  1. public async Task<bool> DoInTranscationAsync(Action<ITransaction> runOperations)
  2. {
  3. var tran = RedisConnection.GenerateConnection.GetDatabase().CreateTransaction();
  4.  
  5. runOperations(tran);
  6.  
  7. return await tran.ExecuteAsync();
  8. }

RedisFactory.cs

  1. using Fantasy.RedisRepository.RedisHelpers;
  2.  
  3. namespace Fantasy.RedisRepository
  4. {
  5. public class RedisFactory
  6. {
  7. /// <summary>
  8. /// 外部访问redis入口,暂时只暴露异步方法
  9. /// </summary>
  10. /// <returns></returns>
  11. public static IRedisHelper CreateRedisRepository()
  12. {
  13. return new RedisHelper();
  14. }
  15. }
  16. }

六. 写在最后

自用RedisRepository分享,仅供参考。有交流才有进步,希望能得到更好的建议,做更好的改进。

如果我的点滴分享对你有点地帮助,欢迎点击下方红色按钮关注,我将持续输出分享。也欢迎为我也为你自己点赞支持。

参考:https://github.com/StackExchange/StackExchange.Redis/tree/master/Docs

                                            ——保持学习,谨记谦虚。不端不装,有趣有梦。

RedisRepository分享和纠错的更多相关文章

  1. 添加Access-Control-Allow-Origin主机头, 授权资源跨站访问

    时间 2014-09-24 22:02:48  All by Neil 原文  https://blog.byneil.com/添加access-control-allow-origin主机头-授权资 ...

  2. Java服务端对Cookie的简单操作

    Java服务端对Cookie的简单操作 时间 2016-04-07 10:39:44 极客头条 原文  http://www.cuiyongzhi.com/index.php/post/15.html ...

  3. StackExchange.Redis帮助类解决方案RedisRepository封装(基础配置)

    本文版权归博客园和作者吴双本人共同所有,转载和爬虫,请注明原文地址.http://www.cnblogs.com/tdws/p/5815735.html 写在前面 这不是教程,分享而已,也欢迎园友们多 ...

  4. STM32下FatFs的移植,实现了坏块管理,硬件ECC,ECC纠错,并进行擦写均衡分析

    最近因项目需要,做一个数据采集的单片机平台.需要移植 FatFs .现在把最后成果贴上来. 1.摘要 在 STM32 单片机上,成功移植 FatFs 0.12b,使用的 Nand Flash 芯片为 ...

  5. 云+社区分享——腾讯云OCR文字识别

    欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文由云+社区运营团队发布在腾讯云+社区 前言 2018年3月27日腾讯云云+社区联合腾讯云智能图像团队共同在客户群举办了腾讯云OCR文字识 ...

  6. webpack4配置详解之常用插件分享

    前言 继上一次webpack的基础配置分享之后,本次将分享一些工作中项目常用的配置插件.也会包含一些自己了解过觉得不错的插件,如有分析不到位的,欢迎纠错,嗯,这些东西文档都有,大佬可绕过. Wepac ...

  7. Nand_ECC_校验和纠错_详解

    word下载 ECC的全称是Error Checking and Correction,是一种用于Nand的差错检测和修正算法.如果操作时序和电路稳定性不存在问题的话,NAND Flash出错的时候一 ...

  8. 封装ShareSDK中的分享功能封以及对类似第三方功能封装的心得【原创】

    本篇的主题有三个: 1.封装思想的介绍 2.我的封装代码 3.我在封装sharesdk(采用的是简洁版本)分享功能是碰到的问题,以及解决方法. PS:其实这个我之前封装过一次,不过最近在重构项目时发现 ...

  9. CTF杂项思路工具分享————2019/5/30

    分享碰到的一些奇奇怪怪的杂项解题方式: 键盘坐标密码: 题目给出一段字符串:11 21 31 18 27 33 34 对照上面的表格,就可以很清晰的看出来密文为:QAZIJCV 猪圈码: 题目为: 一 ...

随机推荐

  1. 控制textbook输入字符

    在KeyPress事件中假如如下代码此实例表示可输入数字退格和“.”. 具体字符KeyChar见连接 http://www.cnblogs.com/linji/archive/2012/10/24/2 ...

  2. CodeForces 614D Skills

    排序+枚举+二分 最大的那些变成A,小的那部分提高最小值 #include<cstdio> #include<cstring> #include<cmath> #i ...

  3. Nagios简介

    Nagios简介 Nagios是一款开源的电脑系统和网络监视工具,能有效监控Windows.Linux和Unix的主机状态,交换机路由器等网络设置,打印机等.在系统或服务状态异常时发出邮件或短信报警第 ...

  4. java SWT嵌入IE,SafeArray .

    java SWT嵌入IE,SafeArray );    // Create a by ref variant    Variant variantByRef = new Variant(pVaria ...

  5. 自动化运维 Ansible

    自动化运维 Ansible 特性 (1).no agents:不需要在被管控主机上安装任何客户端: (2).no server:无服务器端,使用时直接运行命令即可: (3).modules in an ...

  6. WAV音频格式分析

    wav是windows下无损的声音文件,该文件保存了音频的PCM信息和播放器播放该音乐的时候的大多数信息,他有两个块组成,第一个块是格式块,第二个块是数据块 先说格式块,格式块用一张图可以解释,如下 ...

  7. iOS开发之圆角指定

    如果需要将UIView的4个角全部都为圆角,做法相当简单,只需设置其Layer的cornerRadius属性即可(项目需要使用QuartzCore框架).而若要指定某几个角(小于4)为圆角而别的不变时 ...

  8. Jsoup使用教程

    一.解析和遍历一个HTML文档1.解析Html及Url链接 String html = "<html><head><title>First parse&l ...

  9. linux中vi和vim编辑工具

    linux中知名的还有emacs,功能比vim还要强大 vim 如果文件存在vim是打开这个文件,若果不存在,则先新建再打开 命令模式:任何模式都可以通过Esc回到命令模式,命令模式可以通过命令进行选 ...

  10. jQuery学习笔记(二)

    1. 基本选择器 #id      ("#one") .class  $(".two") element (根据给定的元素名匹配元素) $("div& ...