RedisHelper.cs 43 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268
  1. using Newtonsoft.Json;
  2. using StackExchange.Redis;
  3. using System;
  4. using System.Collections.Concurrent;
  5. using System.Collections.Generic;
  6. using System.Linq;
  7. using System.Threading.Tasks;
  8. namespace Masuit.Tools.NoSQL
  9. {
  10. /// <summary>
  11. /// Redis操作
  12. /// </summary>
  13. public class RedisHelper : IDisposable
  14. {
  15. private int DbNum { get; }
  16. private readonly ConnectionMultiplexer _conn;
  17. /// <summary>
  18. /// Redis服务器默认连接字符串,默认为:127.0.0.1:6379,allowadmin=true<br/>
  19. /// </summary>
  20. internal static string RedisConnectionString { get; set; } = "127.0.0.1:6379,allowadmin=true";
  21. /// <summary>
  22. /// 自定义键
  23. /// </summary>
  24. public string CustomKey;
  25. /// <summary>
  26. /// 连接失败 , 如果重新连接成功你将不会收到这个通知
  27. /// </summary>
  28. public event EventHandler<ConnectionFailedEventArgs> ConnectionFailed;
  29. /// <summary>
  30. /// 重新建立连接之前的错误
  31. /// </summary>
  32. public event EventHandler<ConnectionFailedEventArgs> ConnectionRestored;
  33. /// <summary>
  34. /// 发生错误时
  35. /// </summary>
  36. public event EventHandler<RedisErrorEventArgs> ErrorMessage;
  37. /// <summary>
  38. /// 配置更改时
  39. /// </summary>
  40. public event EventHandler<EndPointEventArgs> ConfigurationChanged;
  41. /// <summary>
  42. /// 更改集群时
  43. /// </summary>
  44. public event EventHandler<HashSlotMovedEventArgs> HashSlotMoved;
  45. /// <summary>
  46. /// redis类库错误时
  47. /// </summary>
  48. public event EventHandler<InternalErrorEventArgs> InternalError;
  49. /// <summary>
  50. /// 静态连接池
  51. /// </summary>
  52. private static ConcurrentDictionary<string, ConnectionMultiplexer> ConnectionCache { get; set; } = new ConcurrentDictionary<string, ConnectionMultiplexer>();
  53. #region 构造函数
  54. /// <summary>
  55. /// 构造函数,使用该构造函数需要先在config中配置链接字符串,连接字符串在config配置文件中的ConnectionStrings节下配置,name固定为RedisHosts,值的格式:127.0.0.1:6379,allowadmin=true,若未正确配置,将按默认值“127.0.0.1:6379,allowadmin=true,abortConnect=false”进行操作,如:<br/>
  56. /// &lt;connectionStrings&gt;<br/>
  57. /// &lt;add name = "RedisHosts" connectionString="127.0.0.1:6379,allowadmin=true,abortConnect=false"/&gt;<br/>
  58. /// &lt;/connectionStrings&gt;
  59. /// </summary>
  60. /// <param name="dbNum">数据库编号</param>
  61. public RedisHelper(int dbNum = 0) : this(null, dbNum)
  62. {
  63. }
  64. /// <summary>
  65. /// 构造函数
  66. /// </summary>
  67. /// <param name="readWriteHosts">Redis服务器连接字符串,格式:127.0.0.1:6379,allowadmin=true,abortConnect=false</param>
  68. /// <param name="dbNum">数据库的编号</param>
  69. public RedisHelper(string readWriteHosts, int dbNum = 0)
  70. {
  71. DbNum = dbNum;
  72. _conn = string.IsNullOrWhiteSpace(readWriteHosts) ? ConnectionMultiplexer.Connect(ConfigurationOptions.Parse(RedisConnectionString)) : ConnectionMultiplexer.Connect(ConfigurationOptions.Parse(readWriteHosts));
  73. //_conn.ConfigurationChanged += MuxerConfigurationChanged;
  74. _conn.ConfigurationChanged += ConfigurationChanged;
  75. //_conn.ConnectionFailed += MuxerConnectionFailed;
  76. _conn.ConnectionFailed += ConnectionFailed;
  77. //_conn.ConnectionRestored += MuxerConnectionRestored;
  78. _conn.ConnectionRestored += ConnectionRestored;
  79. //_conn.ErrorMessage += MuxerErrorMessage;
  80. _conn.ErrorMessage += ErrorMessage;
  81. //_conn.HashSlotMoved += MuxerHashSlotMoved;
  82. _conn.HashSlotMoved += HashSlotMoved;
  83. //_conn.InternalError += MuxerInternalError;
  84. _conn.InternalError += InternalError;
  85. }
  86. /// <summary>
  87. /// 构造函数
  88. /// </summary>
  89. /// <param name="readWriteHosts">Redis服务器连接字符串,格式:127.0.0.1:6379,allowadmin=true,abortConnect=false</param>
  90. /// <param name="dbNum">数据库的编号</param>
  91. /// <param name="_"></param>
  92. private RedisHelper(string readWriteHosts, int dbNum, int _)
  93. {
  94. DbNum = dbNum;
  95. readWriteHosts = string.IsNullOrWhiteSpace(readWriteHosts) ? RedisConnectionString : readWriteHosts;
  96. _conn = ConnectionCache.GetOrAdd(readWriteHosts, ConnectionMultiplexer.Connect(ConfigurationOptions.Parse(readWriteHosts)));
  97. //_conn.ConfigurationChanged += MuxerConfigurationChanged;
  98. _conn.ConfigurationChanged += ConfigurationChanged;
  99. //_conn.ConnectionFailed += MuxerConnectionFailed;
  100. _conn.ConnectionFailed += ConnectionFailed;
  101. //_conn.ConnectionRestored += MuxerConnectionRestored;
  102. _conn.ConnectionRestored += ConnectionRestored;
  103. //_conn.ErrorMessage += MuxerErrorMessage;
  104. _conn.ErrorMessage += ErrorMessage;
  105. //_conn.HashSlotMoved += MuxerHashSlotMoved;
  106. _conn.HashSlotMoved += HashSlotMoved;
  107. //_conn.InternalError += MuxerInternalError;
  108. _conn.InternalError += InternalError;
  109. }
  110. /// <summary>
  111. /// 获取新实例
  112. /// </summary>
  113. /// <param name="db">数据库的编号</param>
  114. /// <returns></returns>
  115. public static RedisHelper GetInstance(int db = 0)
  116. {
  117. return new RedisHelper(db);
  118. }
  119. /// <summary>
  120. /// 获取单例
  121. /// </summary>
  122. /// <param name="db">数据库的编号</param>
  123. /// <returns></returns>
  124. public static RedisHelper GetSingleInstance(int db = 0)
  125. {
  126. return new RedisHelper(null, db, 0);
  127. }
  128. /// <summary>
  129. /// 从对象池获取默认实例
  130. /// </summary>
  131. /// <param name="conn">Redis服务器连接字符串,格式:127.0.0.1:6379,allowadmin=true,abortConnect=false</param>
  132. /// <param name="db">数据库的编号</param>
  133. /// <returns></returns>
  134. public static RedisHelper GetInstance(string conn, int db = 0)
  135. {
  136. return new RedisHelper(conn, db);
  137. }
  138. /// <summary>
  139. /// 获取单例
  140. /// </summary>
  141. /// <param name="conn">Redis服务器连接字符串,格式:127.0.0.1:6379,allowadmin=true,abortConnect=false</param>
  142. /// <param name="db">数据库的编号</param>
  143. /// <returns></returns>
  144. public static RedisHelper GetSingleInstance(string conn, int db = 0)
  145. {
  146. return new RedisHelper(conn, db, 0);
  147. }
  148. #endregion 构造函数
  149. #region String
  150. #region 同步方法
  151. /// <summary>
  152. /// 保存单个key value
  153. /// </summary>
  154. /// <param name="key">Redis Key</param>
  155. /// <param name="value">保存的值</param>
  156. /// <param name="expiry">过期时间</param>
  157. /// <returns>是否保存成功</returns>
  158. public bool SetString(string key, string value, TimeSpan? expiry = default(TimeSpan?))
  159. {
  160. key = AddSysCustomKey(key);
  161. return Do(db => db.StringSet(key, value, expiry));
  162. }
  163. /// <summary>
  164. /// 保存多个key value
  165. /// </summary>
  166. /// <param name="keyValues">键值对</param>
  167. /// <returns>是否保存成功</returns>
  168. public bool SetString(List<KeyValuePair<RedisKey, RedisValue>> keyValues)
  169. {
  170. List<KeyValuePair<RedisKey, RedisValue>> newkeyValues = keyValues.Select(p => new KeyValuePair<RedisKey, RedisValue>(AddSysCustomKey(p.Key), p.Value)).ToList();
  171. return Do(db => db.StringSet(newkeyValues.ToArray()));
  172. }
  173. /// <summary>
  174. /// 保存一个对象
  175. /// </summary>
  176. /// <typeparam name="T">对象类型</typeparam>
  177. /// <param name="key">键</param>
  178. /// <param name="obj">值</param>
  179. /// <param name="expiry">过期时间</param>
  180. /// <returns>是否保存成功</returns>
  181. public bool SetString<T>(string key, T obj, TimeSpan? expiry = default(TimeSpan?))
  182. {
  183. key = AddSysCustomKey(key);
  184. string json = ConvertJson(obj);
  185. return Do(db => db.StringSet(key, json, expiry));
  186. }
  187. /// <summary>
  188. /// 获取单个key的值
  189. /// </summary>
  190. /// <param name="key">键</param>
  191. /// <returns>值</returns>
  192. public string GetString(string key)
  193. {
  194. if (KeyExists(key))
  195. {
  196. key = AddSysCustomKey(key);
  197. return Do(db => db.StringGet(key));
  198. }
  199. return string.Empty;
  200. }
  201. /// <summary>
  202. /// 获取多个Key
  203. /// </summary>
  204. /// <param name="listKey">键集合</param>
  205. /// <returns>值集合</returns>
  206. public RedisValue[] GetString(List<string> listKey)
  207. {
  208. List<string> newKeys = listKey.Select(AddSysCustomKey).ToList();
  209. return Do(db => db.StringGet(ConvertRedisKeys(newKeys)));
  210. }
  211. /// <summary>
  212. /// 获取一个key的对象
  213. /// </summary>
  214. /// <typeparam name="T">数据类型</typeparam>
  215. /// <param name="key">键</param>
  216. /// <returns>实例对象</returns>
  217. public T GetString<T>(string key)
  218. {
  219. if (KeyExists(key))
  220. {
  221. key = AddSysCustomKey(key);
  222. return Do(db => ConvertObj<T>(db.StringGet(key)));
  223. }
  224. return default(T);
  225. }
  226. /// <summary>
  227. /// 为数字增长val
  228. /// </summary>
  229. /// <param name="key">键</param>
  230. /// <param name="val">可以为负</param>
  231. /// <returns>增长后的值</returns>
  232. public double StringIncrement(string key, double val = 1)
  233. {
  234. key = AddSysCustomKey(key);
  235. return Do(db => db.StringIncrement(key, val));
  236. }
  237. /// <summary>
  238. /// 为数字减少val
  239. /// </summary>
  240. /// <param name="key">键</param>
  241. /// <param name="val">可以为负</param>
  242. /// <returns>减少后的值</returns>
  243. public double StringDecrement(string key, double val = 1)
  244. {
  245. key = AddSysCustomKey(key);
  246. return Do(db => db.StringDecrement(key, val));
  247. }
  248. #endregion 同步方法
  249. #region 异步方法
  250. /// <summary>
  251. /// 保存单个key value
  252. /// </summary>
  253. /// <param name="key">Redis Key</param>
  254. /// <param name="value">保存的值</param>
  255. /// <param name="expiry">过期时间</param>
  256. /// <returns>是否保存成功</returns>
  257. public async Task<bool> SetStringAsync(string key, string value, TimeSpan? expiry = default(TimeSpan?))
  258. {
  259. key = AddSysCustomKey(key);
  260. return await Do(async db => await db.StringSetAsync(key, value, expiry));
  261. }
  262. /// <summary>
  263. /// 保存多个key value
  264. /// </summary>
  265. /// <param name="keyValues">键值对</param>
  266. /// <returns>是否保存成功</returns>
  267. public async Task<bool> SetStringAsync(List<KeyValuePair<RedisKey, RedisValue>> keyValues)
  268. {
  269. List<KeyValuePair<RedisKey, RedisValue>> newkeyValues = keyValues.Select(p => new KeyValuePair<RedisKey, RedisValue>(AddSysCustomKey(p.Key), p.Value)).ToList();
  270. return await Do(async db => await db.StringSetAsync(newkeyValues.ToArray()));
  271. }
  272. /// <summary>
  273. /// 保存一个对象
  274. /// </summary>
  275. /// <typeparam name="T">数据类型</typeparam>
  276. /// <param name="key">键</param>
  277. /// <param name="obj">需要被缓存的对象</param>
  278. /// <param name="expiry">过期时间</param>
  279. /// <returns>是否保存成功</returns>
  280. public async Task<bool> SetStringAsync<T>(string key, T obj, TimeSpan? expiry = default(TimeSpan?))
  281. {
  282. key = AddSysCustomKey(key);
  283. string json = ConvertJson(obj);
  284. return await Do(async db => await db.StringSetAsync(key, json, expiry));
  285. }
  286. /// <summary>
  287. /// 获取单个key的值
  288. /// </summary>
  289. /// <param name="key">键</param>
  290. /// <returns>值</returns>
  291. public async Task<string> GetStringAsync(string key)
  292. {
  293. if (KeyExists(key))
  294. {
  295. key = AddSysCustomKey(key);
  296. return await Do(async db => await db.StringGetAsync(key));
  297. }
  298. return string.Empty;
  299. }
  300. /// <summary>
  301. /// 获取多个Key
  302. /// </summary>
  303. /// <param name="listKey">键集合</param>
  304. /// <returns>值集合</returns>
  305. public async Task<RedisValue[]> GetStringAsync(List<string> listKey)
  306. {
  307. List<string> newKeys = listKey.Select(AddSysCustomKey).ToList();
  308. return await Do(async db => await db.StringGetAsync(ConvertRedisKeys(newKeys)));
  309. }
  310. /// <summary>
  311. /// 获取一个key的对象
  312. /// </summary>
  313. /// <typeparam name="T">数据类型</typeparam>
  314. /// <param name="key">键</param>
  315. /// <returns>实例对象</returns>
  316. public async Task<T> GetStringAsync<T>(string key)
  317. {
  318. if (KeyExists(key))
  319. {
  320. key = AddSysCustomKey(key);
  321. string result = await Do(async db => await db.StringGetAsync(key));
  322. return ConvertObj<T>(result);
  323. }
  324. return default(T);
  325. }
  326. /// <summary>
  327. /// 为数字增长val
  328. /// </summary>
  329. /// <param name="key">键</param>
  330. /// <param name="val">可以为负</param>
  331. /// <returns>增长后的值</returns>
  332. public async Task<double> IncrementStringAsync(string key, double val = 1)
  333. {
  334. key = AddSysCustomKey(key);
  335. return await Do(async db => await db.StringIncrementAsync(key, val));
  336. }
  337. /// <summary>
  338. /// 为数字减少val
  339. /// </summary>
  340. /// <param name="key">键</param>
  341. /// <param name="val">可以为负</param>
  342. /// <returns>减少后的值</returns>
  343. public async Task<double> DecrementStringAsync(string key, double val = 1)
  344. {
  345. key = AddSysCustomKey(key);
  346. return await Do(async db => await db.StringDecrementAsync(key, val));
  347. }
  348. #endregion 异步方法
  349. #endregion String
  350. #region Hash
  351. #region 同步方法
  352. /// <summary>
  353. /// 判断某个数据是否已经被缓存
  354. /// </summary>
  355. /// <param name="key">键</param>
  356. /// <param name="dataKey">对象的字段</param>
  357. /// <returns>是否缓存成功</returns>
  358. public bool HashExists(string key, string dataKey)
  359. {
  360. key = AddSysCustomKey(key);
  361. return Do(db => db.HashExists(key, dataKey));
  362. }
  363. /// <summary>
  364. /// 存储数据到hash表
  365. /// </summary>
  366. /// <typeparam name="T">数据类型</typeparam>
  367. /// <param name="key">键</param>
  368. /// <param name="dataKey">对象的字段</param>
  369. /// <param name="t">对象实例</param>
  370. /// <returns>是否存储成功</returns>
  371. public bool SetHash<T>(string key, string dataKey, T t)
  372. {
  373. key = AddSysCustomKey(key);
  374. return Do(db =>
  375. {
  376. string json = ConvertJson(t);
  377. return db.HashSet(key, dataKey, json);
  378. });
  379. }
  380. /// <summary>
  381. /// 存储数据到hash表
  382. /// </summary>
  383. /// <typeparam name="T">数据类型</typeparam>
  384. /// <param name="key">键</param>
  385. /// <param name="dataKey">对象的字段</param>
  386. /// <param name="t">对象实例</param>
  387. /// <param name="expire">过期时间</param>
  388. /// <returns>是否存储成功</returns>
  389. public bool SetHash<T>(string key, string dataKey, T t, TimeSpan expire)
  390. {
  391. bool b = SetHash(key, dataKey, t);
  392. Expire(key, expire);
  393. return b;
  394. }
  395. /// <summary>
  396. /// 移除hash中的某值
  397. /// </summary>
  398. /// <param name="key">键</param>
  399. /// <param name="dataKey">对象的字段</param>
  400. /// <returns>是否移除成功</returns>
  401. public bool DeleteHash(string key, string dataKey)
  402. {
  403. key = AddSysCustomKey(key);
  404. return Do(db => db.HashDelete(key, dataKey));
  405. }
  406. /// <summary>
  407. /// 移除hash中的多个值
  408. /// </summary>
  409. /// <param name="key">键</param>
  410. /// <param name="dataKeys">对象的字段集合</param>
  411. /// <returns>数量</returns>
  412. public long DeleteHash(string key, List<RedisValue> dataKeys)
  413. {
  414. key = AddSysCustomKey(key);
  415. //List<RedisValue> dataKeys1 = new List<RedisValue>() {"1","2"};
  416. return Do(db => db.HashDelete(key, dataKeys.ToArray()));
  417. }
  418. /// <summary>
  419. /// 从hash表获取数据
  420. /// </summary>
  421. /// <typeparam name="T">数据类型</typeparam>
  422. /// <param name="key">键</param>
  423. /// <param name="dataKey">对象的字段</param>
  424. /// <returns>对象实例</returns>
  425. public T GetHash<T>(string key, string dataKey)
  426. {
  427. if (KeyExists(key))
  428. {
  429. key = AddSysCustomKey(key);
  430. return Do(db =>
  431. {
  432. string value = db.HashGet(key, dataKey);
  433. return ConvertObj<T>(value);
  434. });
  435. }
  436. return default(T);
  437. }
  438. /// <summary>
  439. /// 为数字增长val
  440. /// </summary>
  441. /// <param name="key">键</param>
  442. /// <param name="dataKey">对象的字段</param>
  443. /// <param name="val">可以为负</param>
  444. /// <returns>增长后的值</returns>
  445. public double IncrementHash(string key, string dataKey, double val = 1)
  446. {
  447. key = AddSysCustomKey(key);
  448. return Do(db => db.HashIncrement(key, dataKey, val));
  449. }
  450. /// <summary>
  451. /// 为数字减少val
  452. /// </summary>
  453. /// <param name="key">键</param>
  454. /// <param name="dataKey">对象的字段</param>
  455. /// <param name="val">可以为负</param>
  456. /// <returns>减少后的值</returns>
  457. public double DecrementHash(string key, string dataKey, double val = 1)
  458. {
  459. key = AddSysCustomKey(key);
  460. return Do(db => db.HashDecrement(key, dataKey, val));
  461. }
  462. /// <summary>
  463. /// 获取hashkey所有Redis key
  464. /// </summary>
  465. /// <typeparam name="T">数据类型</typeparam>
  466. /// <param name="key">键</param>
  467. /// <returns>数据集合</returns>
  468. public List<T> HashKeys<T>(string key)
  469. {
  470. key = AddSysCustomKey(key);
  471. return Do(db =>
  472. {
  473. RedisValue[] values = db.HashKeys(key);
  474. return ConvetList<T>(values);
  475. });
  476. }
  477. #endregion 同步方法
  478. #region 异步方法
  479. /// <summary>
  480. /// 判断某个数据是否已经被缓存
  481. /// </summary>
  482. /// <param name="key">键</param>
  483. /// <param name="dataKey">对象的字段</param>
  484. /// <returns>是否缓存成功</returns>
  485. public async Task<bool> ExistsHashAsync(string key, string dataKey)
  486. {
  487. key = AddSysCustomKey(key);
  488. return await Do(async db => await db.HashExistsAsync(key, dataKey));
  489. }
  490. /// <summary>
  491. /// 存储数据到hash表
  492. /// </summary>
  493. /// <typeparam name="T">数据类型</typeparam>
  494. /// <param name="key">键</param>
  495. /// <param name="dataKey">对象的字段</param>
  496. /// <param name="t">对象实例</param>
  497. /// <returns>是否存储成功</returns>
  498. public async Task<bool> SetHashAsync<T>(string key, string dataKey, T t)
  499. {
  500. key = AddSysCustomKey(key);
  501. return await Do(async db =>
  502. {
  503. string json = ConvertJson(t);
  504. return await db.HashSetAsync(key, dataKey, json);
  505. });
  506. }
  507. /// <summary>
  508. /// 存储数据到hash表
  509. /// </summary>
  510. /// <typeparam name="T">数据类型</typeparam>
  511. /// <param name="key">键</param>
  512. /// <param name="dataKey">对象的字段</param>
  513. /// <param name="t">对象实例</param>
  514. /// <param name="expire">过期时间</param>
  515. /// <returns>是否存储成功</returns>
  516. public async Task<bool> SetHashAsync<T>(string key, string dataKey, T t, TimeSpan expire)
  517. {
  518. var b = await SetHashAsync(key, dataKey, t);
  519. Expire(key, expire);
  520. return b;
  521. }
  522. /// <summary>
  523. /// 移除hash中的某值
  524. /// </summary>
  525. /// <param name="key">键</param>
  526. /// <param name="dataKey">对象的字段</param>
  527. /// <returns>是否移除成功</returns>
  528. public async Task<bool> DeleteHashAsync(string key, string dataKey)
  529. {
  530. key = AddSysCustomKey(key);
  531. return await Do(async db => await db.HashDeleteAsync(key, dataKey));
  532. }
  533. /// <summary>
  534. /// 移除hash中的多个值
  535. /// </summary>
  536. /// <param name="key">键</param>
  537. /// <param name="dataKeys">对象的字段集合</param>
  538. /// <returns>数量</returns>
  539. public async Task<long> DeleteHashAsync(string key, List<RedisValue> dataKeys)
  540. {
  541. key = AddSysCustomKey(key);
  542. return await Do(async db => await db.HashDeleteAsync(key, dataKeys.ToArray()));
  543. }
  544. /// <summary>
  545. /// 从hash表获取数据
  546. /// </summary>
  547. /// <typeparam name="T">数据类型</typeparam>
  548. /// <param name="key">键</param>
  549. /// <param name="dataKey">对象的字段</param>
  550. /// <returns>对象实例</returns>
  551. public async Task<T> GetHashAsync<T>(string key, string dataKey)
  552. {
  553. if (KeyExists(key))
  554. {
  555. key = AddSysCustomKey(key);
  556. string value = await Do(async db => await db.HashGetAsync(key, dataKey));
  557. return ConvertObj<T>(value);
  558. }
  559. return default(T);
  560. }
  561. /// <summary>
  562. /// 为数字增长val
  563. /// </summary>
  564. /// <param name="key">键</param>
  565. /// <param name="dataKey">对象的字段</param>
  566. /// <param name="val">可以为负</param>
  567. /// <returns>增长后的值</returns>
  568. public async Task<double> IncrementHashAsync(string key, string dataKey, double val = 1)
  569. {
  570. key = AddSysCustomKey(key);
  571. return await Do(async db => await db.HashIncrementAsync(key, dataKey, val));
  572. }
  573. /// <summary>
  574. /// 为数字减少val
  575. /// </summary>
  576. /// <param name="key">键</param>
  577. /// <param name="dataKey">对象的字段</param>
  578. /// <param name="val">可以为负</param>
  579. /// <returns>减少后的值</returns>
  580. public async Task<double> DecrementHashAsync(string key, string dataKey, double val = 1)
  581. {
  582. key = AddSysCustomKey(key);
  583. return await Do(async db => await db.HashDecrementAsync(key, dataKey, val));
  584. }
  585. /// <summary>
  586. /// 获取hashkey所有Redis key
  587. /// </summary>
  588. /// <typeparam name="T">数据类型</typeparam>
  589. /// <param name="key">键</param>
  590. /// <returns>数据集合</returns>
  591. public async Task<List<T>> HashKeysAsync<T>(string key)
  592. {
  593. key = AddSysCustomKey(key);
  594. RedisValue[] values = await Do(async db => await db.HashKeysAsync(key));
  595. return ConvetList<T>(values);
  596. }
  597. #endregion 异步方法
  598. #endregion Hash
  599. #region List
  600. #region 同步方法
  601. /// <summary>
  602. /// 移除指定ListId的内部List的值
  603. /// </summary>
  604. /// <typeparam name="T">数据类型</typeparam>
  605. /// <param name="key">键</param>
  606. /// <param name="value">值</param>
  607. public void RemoveList<T>(string key, T value)
  608. {
  609. key = AddSysCustomKey(key);
  610. Do(db => db.ListRemove(key, ConvertJson(value)));
  611. }
  612. /// <summary>
  613. /// 获取指定key的List
  614. /// </summary>
  615. /// <typeparam name="T">数据类型</typeparam>
  616. /// <param name="key">键</param>
  617. /// <returns>数据集</returns>
  618. public List<T> ListRange<T>(string key)
  619. {
  620. if (KeyExists(key))
  621. {
  622. key = AddSysCustomKey(key);
  623. return Do(redis =>
  624. {
  625. var values = redis.ListRange(key);
  626. return ConvetList<T>(values);
  627. });
  628. }
  629. return new List<T>();
  630. }
  631. /// <summary>
  632. /// 入队
  633. /// </summary>
  634. /// <typeparam name="T">数据类型</typeparam>
  635. /// <param name="key">键</param>
  636. /// <param name="value">值</param>
  637. public void ListRightPush<T>(string key, T value)
  638. {
  639. key = AddSysCustomKey(key);
  640. Do(db => db.ListRightPush(key, ConvertJson(value)));
  641. }
  642. /// <summary>
  643. /// 出队
  644. /// </summary>
  645. /// <typeparam name="T">数据类型</typeparam>
  646. /// <param name="key">键</param>
  647. /// <returns>值</returns>
  648. public T ListRightPop<T>(string key)
  649. {
  650. if (KeyExists(key))
  651. {
  652. key = AddSysCustomKey(key);
  653. return Do(db =>
  654. {
  655. var value = db.ListRightPop(key);
  656. return ConvertObj<T>(value);
  657. });
  658. }
  659. return default(T);
  660. }
  661. /// <summary>
  662. /// 入栈
  663. /// </summary>
  664. /// <typeparam name="T">数据类型</typeparam>
  665. /// <param name="key">键</param>
  666. /// <param name="value">值</param>
  667. public void ListLeftPush<T>(string key, T value)
  668. {
  669. key = AddSysCustomKey(key);
  670. Do(db => db.ListLeftPush(key, ConvertJson(value)));
  671. }
  672. /// <summary>
  673. /// 出栈
  674. /// </summary>
  675. /// <typeparam name="T">数据类型</typeparam>
  676. /// <param name="key">键</param>
  677. /// <returns>对象实例</returns>
  678. public T ListLeftPop<T>(string key)
  679. {
  680. if (KeyExists(key))
  681. {
  682. key = AddSysCustomKey(key);
  683. return Do(db =>
  684. {
  685. var value = db.ListLeftPop(key);
  686. return ConvertObj<T>(value);
  687. });
  688. }
  689. return default(T);
  690. }
  691. /// <summary>
  692. /// 获取集合中的数量
  693. /// </summary>
  694. /// <param name="key">键</param>
  695. /// <returns>数量</returns>
  696. public long ListLength(string key)
  697. {
  698. key = AddSysCustomKey(key);
  699. return Do(redis => redis.ListLength(key));
  700. }
  701. #endregion 同步方法
  702. #region 异步方法
  703. /// <summary>
  704. /// 移除指定ListId的内部List的值
  705. /// </summary>
  706. /// <param name="key">键</param>
  707. /// <param name="value">值</param>
  708. public async Task<long> RemoveListAsync<T>(string key, T value)
  709. {
  710. key = AddSysCustomKey(key);
  711. return await Do(async db => await db.ListRemoveAsync(key, ConvertJson(value)));
  712. }
  713. /// <summary>
  714. /// 获取指定key的List
  715. /// </summary>
  716. /// <typeparam name="T">数据类型</typeparam>
  717. /// <param name="key">键</param>
  718. /// <returns>数据集合</returns>
  719. public async Task<List<T>> ListRangeAsync<T>(string key)
  720. {
  721. if (KeyExists(key))
  722. {
  723. key = AddSysCustomKey(key);
  724. var values = await Do(async redis => await redis.ListRangeAsync(key));
  725. return ConvetList<T>(values);
  726. }
  727. return new List<T>();
  728. }
  729. /// <summary>
  730. /// 入队
  731. /// </summary>
  732. /// <typeparam name="T">数据类型</typeparam>
  733. /// <param name="key">键</param>
  734. /// <param name="value">值</param>
  735. public async Task<long> ListRightPushAsync<T>(string key, T value)
  736. {
  737. key = AddSysCustomKey(key);
  738. return await Do(async db => await db.ListRightPushAsync(key, ConvertJson(value)));
  739. }
  740. /// <summary>
  741. /// 出队
  742. /// </summary>
  743. /// <typeparam name="T">数据类型</typeparam>
  744. /// <param name="key">键</param>
  745. /// <returns>对象实例</returns>
  746. public async Task<T> ListRightPopAsync<T>(string key)
  747. {
  748. if (KeyExists(key))
  749. {
  750. key = AddSysCustomKey(key);
  751. var value = await Do(async db => await db.ListRightPopAsync(key));
  752. return ConvertObj<T>(value);
  753. }
  754. return default(T);
  755. }
  756. /// <summary>
  757. /// 入栈
  758. /// </summary>
  759. /// <typeparam name="T">数据类型</typeparam>
  760. /// <param name="key">键</param>
  761. /// <param name="value">值</param>
  762. public async Task<long> ListLeftPushAsync<T>(string key, T value)
  763. {
  764. key = AddSysCustomKey(key);
  765. return await Do(async db => await db.ListLeftPushAsync(key, ConvertJson(value)));
  766. }
  767. /// <summary>
  768. /// 出栈
  769. /// </summary>
  770. /// <typeparam name="T">数据类型</typeparam>
  771. /// <param name="key">键</param>
  772. /// <returns>实例对象</returns>
  773. public async Task<T> ListLeftPopAsync<T>(string key)
  774. {
  775. if (KeyExists(key))
  776. {
  777. key = AddSysCustomKey(key);
  778. var value = await Do(async db => await db.ListLeftPopAsync(key));
  779. return ConvertObj<T>(value);
  780. }
  781. return default(T);
  782. }
  783. /// <summary>
  784. /// 获取集合中的数量
  785. /// </summary>
  786. /// <param name="key">键</param>
  787. /// <returns>数量</returns>
  788. public async Task<long> ListLengthAsync(string key)
  789. {
  790. key = AddSysCustomKey(key);
  791. return await Do(async redis => await redis.ListLengthAsync(key));
  792. }
  793. #endregion 异步方法
  794. #endregion List
  795. #region SortedSet 有序集合
  796. #region 同步方法
  797. /// <summary>
  798. /// 添加
  799. /// </summary>
  800. /// <param name="key">键</param>
  801. /// <param name="value">值</param>
  802. /// <param name="score">排序号</param>
  803. public bool AddSortedSet<T>(string key, T value, double score)
  804. {
  805. key = AddSysCustomKey(key);
  806. return Do(redis => redis.SortedSetAdd(key, ConvertJson<T>(value), score));
  807. }
  808. /// <summary>
  809. /// 删除
  810. /// </summary>
  811. /// <param name="key">键</param>
  812. /// <param name="value">值</param>
  813. public bool RemoveSortedSet<T>(string key, T value)
  814. {
  815. key = AddSysCustomKey(key);
  816. return Do(redis => redis.SortedSetRemove(key, ConvertJson(value)));
  817. }
  818. /// <summary>
  819. /// 获取全部
  820. /// </summary>
  821. /// <param name="key">键</param>
  822. /// <returns>数据集合</returns>
  823. public List<T> SetRangeSortedByRank<T>(string key)
  824. {
  825. if (KeyExists(key))
  826. {
  827. key = AddSysCustomKey(key);
  828. return Do(redis =>
  829. {
  830. var values = redis.SortedSetRangeByRank(key);
  831. return ConvetList<T>(values);
  832. });
  833. }
  834. return new List<T>();
  835. }
  836. /// <summary>
  837. /// 获取集合中的数量
  838. /// </summary>
  839. /// <param name="key">键</param>
  840. /// <returns>数量</returns>
  841. public long SetSortedLength(string key)
  842. {
  843. key = AddSysCustomKey(key);
  844. return Do(redis => redis.SortedSetLength(key));
  845. }
  846. #endregion 同步方法
  847. #region 异步方法
  848. /// <summary>
  849. /// 添加
  850. /// </summary>
  851. /// <param name="key">键</param>
  852. /// <param name="value">值</param>
  853. /// <param name="score">排序号</param>
  854. public async Task<bool> SortedSetAddAsync<T>(string key, T value, double score)
  855. {
  856. key = AddSysCustomKey(key);
  857. return await Do(async redis => await redis.SortedSetAddAsync(key, ConvertJson<T>(value), score));
  858. }
  859. /// <summary>
  860. /// 删除
  861. /// </summary>
  862. /// <param name="key">键</param>
  863. /// <param name="value">值</param>
  864. public async Task<bool> SortedSetRemoveAsync<T>(string key, T value)
  865. {
  866. key = AddSysCustomKey(key);
  867. return await Do(async redis => await redis.SortedSetRemoveAsync(key, ConvertJson(value)));
  868. }
  869. /// <summary>
  870. /// 获取全部
  871. /// </summary>
  872. /// <param name="key">键</param>
  873. /// <returns>数据集合</returns>
  874. public async Task<List<T>> SortedSetRangeByRankAsync<T>(string key)
  875. {
  876. if (KeyExists(key))
  877. {
  878. key = AddSysCustomKey(key);
  879. var values = await Do(async redis => await redis.SortedSetRangeByRankAsync(key));
  880. return ConvetList<T>(values);
  881. }
  882. return new List<T>();
  883. }
  884. /// <summary>
  885. /// 获取集合中的数量
  886. /// </summary>
  887. /// <param name="key">键</param>
  888. /// <returns>数量</returns>
  889. public async Task<long> SortedSetLengthAsync(string key)
  890. {
  891. key = AddSysCustomKey(key);
  892. return await Do(async redis => await redis.SortedSetLengthAsync(key));
  893. }
  894. #endregion 异步方法
  895. #endregion SortedSet 有序集合
  896. #region key
  897. /// <summary>
  898. /// 删除单个key
  899. /// </summary>
  900. /// <param name="key">redis key</param>
  901. /// <returns>是否删除成功</returns>
  902. public bool DeleteKey(string key)
  903. {
  904. key = AddSysCustomKey(key);
  905. return Do(db => db.KeyDelete(key));
  906. }
  907. /// <summary>
  908. /// 删除多个key
  909. /// </summary>
  910. /// <param name="keys">rediskey</param>
  911. /// <returns>成功删除的个数</returns>
  912. public long DeleteKey(List<string> keys)
  913. {
  914. List<string> newKeys = keys.Select(AddSysCustomKey).ToList();
  915. return Do(db => db.KeyDelete(ConvertRedisKeys(newKeys)));
  916. }
  917. /// <summary>
  918. /// 判断key是否存储
  919. /// </summary>
  920. /// <param name="key">键</param>
  921. /// <returns>是否存储成功</returns>
  922. public bool KeyExists(string key)
  923. {
  924. key = AddSysCustomKey(key);
  925. return Do(db => db.KeyExists(key));
  926. }
  927. /// <summary>
  928. /// 重新命名key
  929. /// </summary>
  930. /// <param name="key">旧的键</param>
  931. /// <param name="newKey">新的键</param>
  932. /// <returns>处理结果</returns>
  933. public bool RenameKey(string key, string newKey)
  934. {
  935. key = AddSysCustomKey(key);
  936. return Do(db => db.KeyRename(key, newKey));
  937. }
  938. /// <summary>
  939. /// 设置Key的过期时间
  940. /// </summary>
  941. /// <param name="key">键</param>
  942. /// <param name="expiry">过期时间</param>
  943. /// <returns>处理结果</returns>
  944. public bool Expire(string key, TimeSpan? expiry = default(TimeSpan?))
  945. {
  946. key = AddSysCustomKey(key);
  947. return Do(db => db.KeyExpire(key, expiry));
  948. }
  949. #endregion key
  950. #region 发布订阅
  951. /// <summary>
  952. /// Redis发布订阅 订阅
  953. /// </summary>
  954. /// <param name="subChannel">订阅频道</param>
  955. /// <param name="handler">处理过程</param>
  956. public void Subscribe(string subChannel, Action<RedisChannel, RedisValue> handler = null)
  957. {
  958. ISubscriber sub = _conn.GetSubscriber();
  959. sub.Subscribe(subChannel, (channel, message) =>
  960. {
  961. if (handler == null)
  962. {
  963. Console.WriteLine(subChannel + " 订阅收到消息:" + message);
  964. }
  965. else
  966. {
  967. handler(channel, message);
  968. }
  969. });
  970. }
  971. /// <summary>
  972. /// Redis发布订阅 发布
  973. /// </summary>
  974. /// <typeparam name="T">消息对象</typeparam>
  975. /// <param name="channel">发布频道</param>
  976. /// <param name="msg">消息</param>
  977. /// <returns>消息的数量</returns>
  978. public long Publish<T>(string channel, T msg)
  979. {
  980. ISubscriber sub = _conn.GetSubscriber();
  981. return sub.Publish(channel, ConvertJson(msg));
  982. }
  983. /// <summary>
  984. /// Redis发布订阅 取消订阅
  985. /// </summary>
  986. /// <param name="channel">频道</param>
  987. public void Unsubscribe(string channel)
  988. {
  989. ISubscriber sub = _conn.GetSubscriber();
  990. sub.Unsubscribe(channel);
  991. }
  992. /// <summary>
  993. /// Redis发布订阅 取消全部订阅
  994. /// </summary>
  995. public void UnsubscribeAll()
  996. {
  997. ISubscriber sub = _conn.GetSubscriber();
  998. sub.UnsubscribeAll();
  999. }
  1000. #endregion 发布订阅
  1001. #region 其他
  1002. /// <summary>
  1003. /// 创建一个事务
  1004. /// </summary>
  1005. /// <returns>事务对象</returns>
  1006. public ITransaction CreateTransaction()
  1007. {
  1008. return GetDatabase().CreateTransaction();
  1009. }
  1010. /// <summary>
  1011. /// 获得一个数据库实例
  1012. /// </summary>
  1013. /// <returns>数据库实例</returns>
  1014. public IDatabase GetDatabase()
  1015. {
  1016. return _conn.GetDatabase(DbNum);
  1017. }
  1018. /// <summary>
  1019. /// 获得一个服务器实例
  1020. /// </summary>
  1021. /// <param name="hostAndPort">服务器地址</param>
  1022. /// <returns>服务器实例</returns>
  1023. public IServer GetServer(string hostAndPort = null)
  1024. {
  1025. hostAndPort = string.IsNullOrEmpty(hostAndPort) ? _conn.Configuration.Split(',')[0] : hostAndPort;
  1026. return _conn.GetServer(hostAndPort);
  1027. }
  1028. /// <summary>
  1029. /// 设置前缀
  1030. /// </summary>
  1031. /// <param name="customKey">前缀</param>
  1032. public void SetSysCustomKey(string customKey)
  1033. {
  1034. CustomKey = customKey;
  1035. }
  1036. #endregion 其他
  1037. #region 辅助方法
  1038. private string AddSysCustomKey(string oldKey)
  1039. {
  1040. var prefixKey = CustomKey ?? String.Empty;
  1041. return prefixKey + oldKey;
  1042. }
  1043. private T Do<T>(Func<IDatabase, T> func)
  1044. {
  1045. var database = _conn.GetDatabase(DbNum);
  1046. return func(database);
  1047. }
  1048. private string ConvertJson<T>(T value)
  1049. {
  1050. return JsonConvert.SerializeObject(value);
  1051. }
  1052. private T ConvertObj<T>(RedisValue value)
  1053. {
  1054. return JsonConvert.DeserializeObject<T>(value);
  1055. }
  1056. private List<T> ConvetList<T>(RedisValue[] values)
  1057. {
  1058. List<T> result = new List<T>();
  1059. foreach (var item in values)
  1060. {
  1061. var model = ConvertObj<T>(item);
  1062. result.Add(model);
  1063. }
  1064. return result;
  1065. }
  1066. private RedisKey[] ConvertRedisKeys(List<string> redisKeys)
  1067. {
  1068. return redisKeys.Select(redisKey => (RedisKey)redisKey).ToArray();
  1069. }
  1070. #endregion 辅助方法
  1071. #region 事件
  1072. ///// <summary>
  1073. ///// 配置更改时
  1074. ///// </summary>
  1075. ///// <param name="sender">触发者</param>
  1076. ///// <param name="e">事件参数</param>
  1077. //private static void MuxerConfigurationChanged(object sender, EndPointEventArgs e)
  1078. //{
  1079. // Console.WriteLine("Configuration changed: " + e.EndPoint);
  1080. //}
  1081. ///// <summary>
  1082. ///// 发生错误时
  1083. ///// </summary>
  1084. ///// <param name="sender"></param>
  1085. ///// <param name="e"></param>
  1086. //private static void MuxerErrorMessage(object sender, RedisErrorEventArgs e)
  1087. //{
  1088. // Console.WriteLine("ErrorMessage: " + e.Message);
  1089. //}
  1090. ///// <summary>
  1091. ///// 重新建立连接之前的错误
  1092. ///// </summary>
  1093. ///// <param name="sender"></param>
  1094. ///// <param name="e"></param>
  1095. //private static void MuxerConnectionRestored(object sender, ConnectionFailedEventArgs e)
  1096. //{
  1097. // Console.WriteLine("ConnectionRestored: " + e.EndPoint);
  1098. //}
  1099. ///// <summary>
  1100. ///// 连接失败 , 如果重新连接成功你将不会收到这个通知
  1101. ///// </summary>
  1102. ///// <param name="sender"></param>
  1103. ///// <param name="e"></param>
  1104. //private static void MuxerConnectionFailed(object sender, ConnectionFailedEventArgs e)
  1105. //{
  1106. // Console.WriteLine("重新连接:Endpoint failed: " + e.EndPoint + ", " + e.FailureType + (e.Exception == null ? "" : (", " + e.Exception.Message)));
  1107. //}
  1108. ///// <summary>
  1109. ///// 更改集群
  1110. ///// </summary>
  1111. ///// <param name="sender"></param>
  1112. ///// <param name="e"></param>
  1113. //private static void MuxerHashSlotMoved(object sender, HashSlotMovedEventArgs e)
  1114. //{
  1115. // Console.WriteLine("HashSlotMoved:NewEndPoint" + e.NewEndPoint + ", OldEndPoint" + e.OldEndPoint);
  1116. //}
  1117. ///// <summary>
  1118. ///// redis类库错误
  1119. ///// </summary>
  1120. ///// <param name="sender"></param>
  1121. ///// <param name="e"></param>
  1122. //private static void MuxerInternalError(object sender, InternalErrorEventArgs e)
  1123. //{
  1124. // Console.WriteLine("InternalError:Message" + e.Exception.Message);
  1125. //}
  1126. #endregion 事件
  1127. /// <summary>Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources.</summary>
  1128. public void Dispose()
  1129. {
  1130. _conn.Dispose();
  1131. }
  1132. }
  1133. }