RedisHelper.cs 43 KB

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