日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > C# >内容正文

C#

c#进阶(4)—— Redis 用于消息队列的存储

發布時間:2023/12/10 C# 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 c#进阶(4)—— Redis 用于消息队列的存储 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1、參考的博文

a : http://www.cnblogs.com/lori/archive/2012/04/12/2443708.html —— 主要的實現思路

b:? http://www.cnblogs.com/liqingwen/archive/2017/04/06/6672452.html —— RedisHelper 類

c : https://www.cnblogs.com/stopfalling/p/5375492.html —— 應用場景說明

2、原理說明

??? 博文a 中的老師,提供了Redis 實現消息隊列的整體思路,言簡意賅,但部分類庫a 老師并未提供,因此我參照了博文b 中老師的RedisHelper 類,主要借鑒的方法為ListLeftPop及ListRightPush,及實現消息隊列的核心思想,先進先出。

??? 博文c 中老師詳細介紹了幾種消息隊列的 業務場景,是我所看的所有業務場景描述中最為詳細,清晰的,結合博文a 老師的整體思路,對消息隊列的實現上有了較為清晰的認識。

整體思路在博文a 中老師已經介紹,及依靠 mvc 框架,通過web端用戶提交事件,實現消息入列,通過定時器(Timer)實現按照時間間隔的消息出列,Redis 作為存儲媒介,存儲消

息內容。

3、具體實現

3.1 前期準備

(1)開發工具:VS 2017;redis-desktop-manager 客戶端;Redis版本:Redis-x64-3.2.100

(2)新建MVC5 框架,配置 web.config 文件,RouteConfig文件

?????? a、新建MVC5 框架

???????? b、配置web.config 文件(Default key 主要用于區別不同的key值,此處使用了解決方法名稱)

  ? c、配置路由(使用經典模式即可,路由的詳細配置說明可參見我的另一篇博文 )

  

(3)目前.net 框架中支持Redis 的dll包括ServiceStackRedis 及 StackExchange,前者下載地址在Git hub 上,但該類庫在6000并發后會拋出license exception 的異常。因此

我此次使用的dll 為基于微軟的StackExchage dll,該dll的下載方式是,在NuGet中,找到StackExchage.Redis 類包,下載即可,見下圖。

(4)創建消息實體對象,如下圖所示

?

(此處userId 為 字符型,int 型在序列化時,會出現異常,請注意。)

?3.2 引入Redis 類庫,此部分未做大量修改,代碼拷貝自 博文b?

using System; using System.Linq; using StackExchange.Redis; using System.Configuration; using System.Collections.Generic; using System.Runtime.Serialization.Formatters.Binary; using System.IO; using System.Threading.Tasks; using System.Runtime.Serialization;namespace MVC5Project.Redis.BaseOnStackExchage {public class RedisHelper{/// <summary>/// 連接字符串/// </summary>private static readonly string ConnectionString;/// <summary>/// redis 連接對象/// </summary>private static IConnectionMultiplexer _connMultiplexer;/// <summary>/// 默認的key值(用來當作RedisKey的前綴)【此部分為自行修改的,無意義】/// </summary>public static string DefaultKey { get; private set; }/// <summary>/// 鎖/// </summary>private static readonly object Locker = new object();/// <summary>/// 數據庫訪問對象/// </summary>private readonly IDatabase _db;/// <summary>/// 采用雙重鎖單例模式,保證數據訪問對象有且僅有一個/// </summary>/// <returns></returns>public IConnectionMultiplexer GetConnectionRedisMultiplexer(){if ((_connMultiplexer == null || !_connMultiplexer.IsConnected)){lock (Locker){if ((_connMultiplexer == null || !_connMultiplexer.IsConnected)){_connMultiplexer = ConnectionMultiplexer.Connect(ConnectionString);}}}return _connMultiplexer;}/// <summary>/// 添加事務處理/// </summary>/// <returns></returns>public ITransaction GetTransaction(){//創建事務return _db.CreateTransaction();}/// <summary>/// 靜態的構造函數,/// 構造函數是屬于類的,而不是屬于實例的/// 就是說這個構造函數只會被執行一次。也就是在創建第一個實例或引用任何靜態成員之前,由.NET自動調用。/// </summary>static RedisHelper(){ConnectionString = ConfigurationManager.ConnectionStrings["RedisConnectionString"].ConnectionString;_connMultiplexer = ConnectionMultiplexer.Connect(ConnectionString);DefaultKey = ConfigurationManager.AppSettings["Redis.DefaultKey"];RegisterEvent();}/// <summary>/// 重載構造器/// </summary>/// <param name="db"></param>public RedisHelper(int db = -1){_db = _connMultiplexer.GetDatabase(db);}#region private method/// <summary>/// 添加 key 的前綴/// </summary>/// <param name="key"></param>/// <returns></returns>private static string AddKeyPrefix(string key){return $"{DefaultKey}:{key}";}/// <summary>/// 序列化/// </summary>/// <param name="obj"></param>/// <returns></returns>private static byte[] Serialize(object obj){try{if (obj == null)return null;var binaryFormatter = new BinaryFormatter();using (var memoryStream = new MemoryStream()){binaryFormatter.Serialize(memoryStream, obj);var data = memoryStream.ToArray();return data;}}catch (SerializationException ex) {throw ex;}}/// <summary>/// 反序列化/// </summary>/// <typeparam name="T"></typeparam>/// <param name="data"></param>/// <returns></returns>private static T Deserialize<T>(byte[] data){if (data == null)return default(T);var binaryFormatter = new BinaryFormatter();using (var memoryStream = new MemoryStream(data)){var result = (T)binaryFormatter.Deserialize(memoryStream);return result;}}#endregion #region stringGet /// <summary>/// 設置key,并保存字符串(如果key 已存在,則覆蓋)/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <param name="expried"></param>/// <returns></returns>public bool StringSet(string redisKey, string redisValue, TimeSpan? expried = null){redisKey = AddKeyPrefix(redisKey);return _db.StringSet(redisKey, redisValue, expried);}/// <summary>/// 保存多個key-value/// </summary>/// <param name="keyValuePairs"></param>/// <returns></returns>public bool StringSet(IEnumerable<KeyValuePair<RedisKey, RedisValue>> keyValuePairs){keyValuePairs =keyValuePairs.Select(x => new KeyValuePair<RedisKey, RedisValue>(AddKeyPrefix(x.Key), x.Value));return _db.StringSet(keyValuePairs.ToArray());}/// <summary>/// 獲取字符串/// </summary>/// <param name="redisKey"></param>/// <param name="expired"></param>/// <returns></returns>public string StringGet(string redisKey, TimeSpan? expired = null){try{redisKey = AddKeyPrefix(redisKey);return _db.StringGet(redisKey);}catch (TypeAccessException ex){throw ex;}}/// <summary>/// 存儲一個對象,該對象會被序列化存儲/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <param name="expired"></param>/// <returns></returns>public bool StringSet<T>(string redisKey, T redisValue, TimeSpan? expired = null){redisKey = AddKeyPrefix(redisKey);var json = Serialize(redisKey);return _db.StringSet(redisKey, json, expired);}/// <summary>/// 獲取一個對象(會進行反序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="expired"></param>/// <returns></returns>public T StringSet<T>(string redisKey, TimeSpan? expired = null){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(_db.StringGet(redisKey));}/// <summary>/// 保存一個字符串值/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <param name="expired"></param>/// <returns></returns>public async Task<bool> StringSetAsync(string redisKey, string redisValue, TimeSpan? expired = null){redisKey = AddKeyPrefix(redisKey);return await _db.StringSetAsync(redisKey, redisValue, expired);}/// <summary>/// 保存一個字符串值/// </summary>/// <param name="keyValuePairs"></param>/// <returns></returns>public async Task<bool> StringSetAsync(IEnumerable<KeyValuePair<RedisKey, RedisValue>> keyValuePairs){keyValuePairs= keyValuePairs.Select(x => new KeyValuePair<RedisKey, RedisValue>(AddKeyPrefix(x.Key), x.Value));return await _db.StringSetAsync(keyValuePairs.ToArray());}/// <summary>/// 獲取單個值/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <param name="expired"></param>/// <returns></returns>public async Task<string> StringGetAsync(string redisKey, string redisValue, TimeSpan? expired = null){redisKey = AddKeyPrefix(redisKey);return await _db.StringGetAsync(redisKey);}/// <summary>/// 存儲一個對象(該對象會被序列化保存)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <param name="expired"></param>/// <returns></returns>public async Task<bool> StringSetAsync<T>(string redisKey, string redisValue, TimeSpan? expired = null){redisKey = AddKeyPrefix(redisKey);var json = Serialize(redisValue);return await _db.StringSetAsync(redisKey, json, expired);}/// <summary>/// 獲取一個對象(反序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <param name="expired"></param>/// <returns></returns>public async Task<T> StringGetAsync<T>(string redisKey, string redisValue, TimeSpan? expired = null){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(await _db.StringGetAsync(redisKey));}#endregion#region string operation/// <summary>/// 判斷字段是否在hash中/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public bool HashExist(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return _db.HashExists(redisKey, hashField);}/// <summary>/// 從hash 中刪除字段/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public bool HashDelete(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return _db.HashDelete(redisKey, hashField);}/// <summary>/// 從hash中移除指定字段/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public long HashDelete(string redisKey, IEnumerable<RedisValue> hashField){redisKey = AddKeyPrefix(redisKey);return _db.HashDelete(redisKey, hashField.ToArray());}/// <summary>/// 在hash中設定值/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <param name="value"></param>/// <returns></returns>public bool HashSet(string redisKey, string hashField, string value){redisKey = AddKeyPrefix(redisKey);return _db.HashSet(redisKey, hashField, value);}/// <summary>/// 從Hash 中獲取值/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public RedisValue HashGet(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return _db.HashGet(redisKey, hashField);}/// <summary>/// 從Hash 中獲取值/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public RedisValue[] HashGet(string redisKey, RedisValue[] hashField){redisKey = AddKeyPrefix(redisKey);return _db.HashGet(redisKey, hashField);}/// <summary>/// 從hash 返回所有的key值/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public IEnumerable<RedisValue> HashKeys(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.HashKeys(redisKey);}/// <summary>/// 根據key返回hash中的值/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public RedisValue[] HashValues(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.HashValues(redisKey);}/// <summary>/// /// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <param name="value"></param>/// <returns></returns>public bool HashSet<T>(string redisKey, string hashField, T value){redisKey = AddKeyPrefix(redisKey);var json = Serialize(value);return _db.HashSet(redisKey, hashField, json);}/// <summary>/// 在hash 中獲取值 (反序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public T HashGet<T>(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(_db.HashGet(redisKey, hashField));}/// <summary>/// 判斷字段是否存在hash 中/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public async Task<bool> HashExistsAsync(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return await _db.HashExistsAsync(redisKey, hashField);}/// <summary>/// 從hash中移除指定字段/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public async Task<bool> HashDeleteAsync(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return await _db.HashDeleteAsync(redisKey, hashField);}/// <summary>/// 從hash中移除指定字段/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public async Task<long> HashDeleteAsync(string redisKey, IEnumerable<RedisValue> hashField){redisKey = AddKeyPrefix(redisKey);return await _db.HashDeleteAsync(redisKey, hashField.ToArray());}/// <summary>/// 在hash 設置值/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <param name="value"></param>/// <returns></returns>public async Task<bool> HashSetAsync(string redisKey, string hashField, string value){redisKey = AddKeyPrefix(redisKey);return await _db.HashSetAsync(redisKey, hashField, value);}/// <summary>/// 在hash 中設定值/// </summary>/// <param name="redisKey"></param>/// <param name="hashFields"></param>/// <returns></returns>public async Task HashSetAsync(string redisKey, IEnumerable<HashEntry> hashFields){redisKey = AddKeyPrefix(redisKey);await _db.HashSetAsync(redisKey, hashFields.ToArray());}/// <summary>/// 在hash 中設定值/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public async Task<RedisValue> HashGetAsync(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return await _db.HashGetAsync(redisKey, hashField);}/// <summary>/// 在hash 中獲取值/// </summary>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <param name="value"></param>/// <returns></returns>public async Task<IEnumerable<RedisValue>> HashGetAsync(string redisKey, RedisValue[] hashField, string value){redisKey = AddKeyPrefix(redisKey);return await _db.HashGetAsync(redisKey, hashField);}/// <summary>/// 從hash返回所有的字段值/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<IEnumerable<RedisValue>> HashKeysAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.HashKeysAsync(redisKey);}/// <summary>/// 返回hash中所有的值/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<IEnumerable<RedisValue>> HashValuesAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.HashValuesAsync(redisKey);}/// <summary>/// 在hash 中設定值(序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <param name="value"></param>/// <returns></returns>public async Task<bool> HashSetAsync<T>(string redisKey, string hashField, T value){redisKey = AddKeyPrefix(redisKey);var json = Serialize(value);return await _db.HashSetAsync(redisKey, hashField, json);}/// <summary>/// 在hash中獲取值(反序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="hashField"></param>/// <returns></returns>public async Task<T> HashGetAsync<T>(string redisKey, string hashField){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(await _db.HashGetAsync(redisKey, hashField));}#endregion#region list operation/// <summary>/// 移除并返回key所對應列表的第一個元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public string ListLeftPop(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.ListLeftPop(redisKey);}/// <summary>/// 移除并返回key所對應列表的最后一個元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public string ListRightPop(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.ListRightPop(redisKey);}/// <summary>/// 移除指定key及key所對應的元素/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public long ListRemove(string redisKey, string redisValue){redisKey = AddKeyPrefix(redisKey);return _db.ListRemove(redisKey, redisValue);}/// <summary>/// 在列表尾部插入值,如果鍵不存在,先創建再插入值/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public long ListRightPush(string redisKey, string redisValue){redisKey = AddKeyPrefix(redisKey);return _db.ListRightPush(redisKey, redisValue);}/// <summary>/// 在列表頭部插入值,如果鍵不存在,先創建再插入值/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public long ListLeftPush(string redisKey, string redisValue){redisKey = AddKeyPrefix(redisKey);return _db.ListLeftPush(redisKey, redisValue);}/// <summary>/// 返回列表上該鍵的長度,如果不存在,返回0/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public long ListLength(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.ListLength(redisKey);}/// <summary>/// 返回在該列表上鍵所對應的元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public IEnumerable<RedisValue> ListRange(string redisKey){try{redisKey = AddKeyPrefix(redisKey);return _db.ListRange(redisKey);}catch (Exception ex){throw ex;}}/// <summary>/// 移除并返回存儲在該鍵列表的第一個元素/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <returns></returns>public T ListLeftPop<T>(string redisKey){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(_db.ListLeftPop(redisKey));}/// <summary>/// 移除并返回該列表上的最后一個元素/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <returns></returns>public T ListRightPop<T>(string redisKey){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(_db.ListRightPop(redisKey));}/// <summary>/// 在列表尾部插入值,如果鍵不存在,先創建再插入值/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public long ListRightPush<T>(string redisKey, T redisValue){redisKey = AddKeyPrefix(redisKey);return _db.ListRightPush(redisKey, Serialize(redisValue));}/// <summary>/// 在列表頭部插入值,如果鍵不存在,創建后插入值/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public long ListLeftPush<T>(string redisKey, T redisValue){redisKey = AddKeyPrefix(redisKey);return _db.ListRightPush(redisKey, Serialize(redisValue));}/// <summary>/// 移除并返回存儲在該鍵列表的第一個元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<string> ListLeftPopAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.ListLeftPopAsync(redisKey);}/// <summary>/// 移除并返回存儲在該鍵列表的最后一個元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<string> ListRightPopAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.ListRightPopAsync(redisKey);}/// <summary>/// 移除列表指定鍵上與值相同的元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<long> ListRemoveAsync(string redisKey,string redisValue){redisKey = AddKeyPrefix(redisKey);return await _db.ListRemoveAsync(redisKey, redisValue);}/// <summary>/// 在列表尾部差入值,如果鍵不存在,先創建后插入/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public async Task<long> ListRightPushAsync(string redisKey,string redisValue){redisKey = AddKeyPrefix(redisKey);return await ListRightPushAsync(redisKey, redisValue);}/// <summary>/// 在列表頭部插入值,如果鍵不存在,先創建后插入/// </summary>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public async Task<long> ListLeftPushAsync(string redisKey,string redisValue){redisKey = AddKeyPrefix(redisKey);return await _db.ListLeftPushAsync(redisKey, redisValue);}/// <summary>/// 返回列表上的長度,如果不存在,返回0/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<long> ListLengthAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.ListLengthAsync(redisKey);}/// <summary>/// 返回在列表上鍵對應的元素/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<IEnumerable<RedisValue>> ListRangeAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.ListRangeAsync(redisKey);}/// <summary>/// 移除并返回存儲在key對應列表的第一個元素/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <returns></returns>public async Task<T> ListLeftPopAsync<T>(string redisKey){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(await _db.ListLeftPopAsync(redisKey));}/// <summary>/// 移除并返回存儲在key 對應列表的最后一個元素/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <returns></returns>public async Task<T> ListRightPopAsync<T>(string redisKey){redisKey = AddKeyPrefix(redisKey);return Deserialize<T>(await _db.ListRightPopAsync(redisKey));}/// <summary>/// 在列表尾部插入值,如果值不存在,先創建后寫入值/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public async Task<long> ListRightPushAsync<T>(string redisKey,string redisValue){redisKey = AddKeyPrefix(redisKey);return await _db.ListRightPushAsync(redisKey, Serialize(redisValue));}/// <summary>/// 在列表頭部插入值,如果值不存在,先創建后寫入值/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="redisValue"></param>/// <returns></returns>public async Task<long> ListLeftPushAsync<T>(string redisKey,string redisValue){redisKey = AddKeyPrefix(redisKey);return await _db.ListLeftPushAsync(redisKey, Serialize(redisValue));}#endregion#region sorted set operation/// <summary>/// sortedset 新增/// </summary>/// <param name="redisKey"></param>/// <param name="member"></param>/// <param name="score"></param>/// <returns></returns>public bool SortedSetAdd(string redisKey,string member,double score){redisKey = AddKeyPrefix(redisKey);return _db.SortedSetAdd(redisKey, member, score);}/// <summary>/// 在有序集合中返回指定范圍的元素,默認情況下由低到高/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public IEnumerable<RedisValue> SortedSetRangeByRank(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.SortedSetRangeByRank(redisKey);}/// <summary>/// 返回有序集合的個數/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public long SortedSetLength(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.SortedSetLength(redisKey);}/// <summary>/// 返回有序集合的元素個數/// </summary>/// <param name="redisKey"></param>/// <param name="member"></param>/// <returns></returns>public bool SortedSetLength(string redisKey,string member){redisKey = AddKeyPrefix(redisKey);return _db.SortedSetRemove(redisKey, member);}/// <summary>/// sorted set Add/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="member"></param>/// <param name="score"></param>/// <returns></returns>public bool SortedSetAdd<T>(string redisKey,T member,double score){redisKey = AddKeyPrefix(redisKey);var json = Serialize(member);return _db.SortedSetAdd(redisKey, json, score);}/// <summary>/// sorted set add/// </summary>/// <param name="redisKey"></param>/// <param name="member"></param>/// <param name="score"></param>/// <returns></returns>public async Task<bool> SortedSetAddAsync(string redisKey,string member,double score){redisKey = AddKeyPrefix(redisKey);return await _db.SortedSetAddAsync(redisKey, member, score);}/// <summary>/// 在有序集合中返回指定范圍的元素,默認情況下由低到高/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<IEnumerable<RedisValue>> SortedSetRangeByRankAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.SortedSetRangeByRankAsync(redisKey);}/// <summary>/// 返回有序集合的元素個數/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<long> SortedSetLengthAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.SortedSetLengthAsync(redisKey);}/// <summary>/// 返回有序集合的元素個數/// </summary>/// <param name="redisKey"></param>/// <param name="member"></param>/// <returns></returns>public async Task<bool> SortedSetRemoveAsync(string redisKey,string member){redisKey = AddKeyPrefix(redisKey);return await _db.SortedSetRemoveAsync(redisKey, member);}/// <summary>/// SortedSet 新增/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisKey"></param>/// <param name="member"></param>/// <param name="score"></param>/// <returns></returns>public async Task<bool> SortedSetAddAsync<T>(string redisKey,T member,double score){redisKey = AddKeyPrefix(redisKey);var json = Serialize(member);return await _db.SortedSetAddAsync(redisKey, json, score);}#endregion#region key operation/// <summary>/// 移除指定key/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public bool KeyDelete(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.KeyDelete(redisKey);}/// <summary>/// 刪除指定key/// </summary>/// <param name="redisKeys"></param>/// <returns></returns>public long KeyDelete(IEnumerable<string> redisKeys){var keys = redisKeys.Select(x => (RedisKey)AddKeyPrefix(x));return _db.KeyDelete(keys.ToArray());}/// <summary>/// 檢驗key是否存在/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public bool KeyExists(string redisKey){redisKey = AddKeyPrefix(redisKey);return _db.KeyExists(redisKey);}/// <summary>/// 重命名key/// </summary>/// <param name="oldKeyName"></param>/// <param name="newKeyName"></param>/// <returns></returns>public bool KeyReName(string oldKeyName,string newKeyName){oldKeyName = AddKeyPrefix(oldKeyName);return _db.KeyRename(oldKeyName, newKeyName);}/// <summary>/// 設置key 的過期時間/// </summary>/// <param name="redisKey"></param>/// <param name="expired"></param>/// <returns></returns>public bool KeyExpire(string redisKey,TimeSpan?expired = null){redisKey = AddKeyPrefix(redisKey);return _db.KeyExpire(redisKey, expired);}/// <summary>/// 移除指定的key/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<bool> KeyDeleteAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.KeyDeleteAsync(redisKey);}/// <summary>/// 刪除指定的key/// </summary>/// <param name="redisKeys"></param>/// <returns></returns>public async Task<long> KeyDeleteAsync(IEnumerable<string> redisKeys){var keys = redisKeys.Select(x => (RedisKey)AddKeyPrefix(x));return await _db.KeyDeleteAsync(keys.ToArray());}/// <summary>/// 檢驗key 是否存在/// </summary>/// <param name="redisKey"></param>/// <returns></returns>public async Task<bool> KeyExistsAsync(string redisKey){redisKey = AddKeyPrefix(redisKey);return await _db.KeyExistsAsync(redisKey);}/// <summary>/// 重命名key/// </summary>/// <param name="redisKey"></param>/// <param name="redisNewKey"></param>/// <returns></returns>public async Task<bool> KeyRenameAsync(string redisKey,string redisNewKey){redisKey = AddKeyPrefix(redisKey);return await _db.KeyRenameAsync(redisKey, redisNewKey);}/// <summary>/// 設置 key 時間/// </summary>/// <param name="redisKey"></param>/// <param name="expired"></param>/// <returns></returns>public async Task<bool> KeyExpireAsync(string redisKey,TimeSpan? expired){redisKey = AddKeyPrefix(redisKey);return await _db.KeyExpireAsync(redisKey, expired);}#endregion#region Subscribe/// <summary>/// 訂閱/// </summary>/// <param name="channel">頻道</param>/// <param name="handle">事件</param>public void Subscribe(RedisChannel channel,Action<RedisChannel,RedisValue> handle){//getSubscriber() 獲取到指定服務器的發布者訂閱者的連接var sub = _connMultiplexer.GetSubscriber();//訂閱執行某些操作時改變了 優先/主動 節點廣播sub.Subscribe(channel, handle);}/// <summary>/// 發布/// </summary>/// <param name="channel"></param>/// <param name="message"></param>/// <returns></returns>public long Publish(RedisChannel channel,RedisValue message){var sub = _connMultiplexer.GetSubscriber();return sub.Publish(channel, message);}/// <summary>/// 發布(使用序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="channel"></param>/// <param name="message"></param>/// <returns></returns>public long Publish<T>(RedisChannel channel,T message){var sub = _connMultiplexer.GetSubscriber();return sub.Publish(channel, Serialize(message));}/// <summary>/// 訂閱/// </summary>/// <param name="redisChannel"></param>/// <param name="handle"></param>/// <returns></returns>public async Task SubscribeAsync(RedisChannel redisChannel,Action<RedisChannel,RedisValue> handle){var sub = _connMultiplexer.GetSubscriber();await sub.SubscribeAsync(redisChannel, handle);}/// <summary>/// 發布/// </summary>/// <param name="redisChannel"></param>/// <param name="message"></param>/// <returns></returns>public async Task<long> PublishAsync(RedisChannel redisChannel,RedisValue message){var sub = _connMultiplexer.GetSubscriber();return await sub.PublishAsync(redisChannel, message);}/// <summary>/// 發布(使用序列化)/// </summary>/// <typeparam name="T"></typeparam>/// <param name="redisChannel"></param>/// <param name="message"></param>/// <returns></returns>public async Task<long> PublishAsync<T>(RedisChannel redisChannel,T message){var sub = _connMultiplexer.GetSubscriber();return await sub.PublishAsync(redisChannel, Serialize(message));}#endregion#region register event/// <summary>/// 注冊事件/// </summary>private static void RegisterEvent(){_connMultiplexer.ConnectionRestored += ConnMultiplexer_ConnectionRestored;_connMultiplexer.ConnectionFailed += ConnMultiplexer_ConnectionFailed;_connMultiplexer.ErrorMessage += ConnMultiplexer_ErrorMessage;_connMultiplexer.ConfigurationChanged += ConnMultiplexer_ConfigurationChanged;_connMultiplexer.HashSlotMoved += ConnMultiplexer_HashSlotMoved;_connMultiplexer.InternalError += ConnMultiplexer_InternalError;_connMultiplexer.ConfigurationChangedBroadcast += ConnMultiplexer_ConfigurationChangedBroadcast;}/// <summary>/// 重新配置廣播時(主從同步更改)/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_ConfigurationChangedBroadcast(object sender, EndPointEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_ConfigurationChangedBroadcast)}: {e.EndPoint}");}/// <summary>/// 發生內部錯誤時(調試用)/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_InternalError(object sender,InternalErrorEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_InternalError)}: {e.Exception}");}/// <summary>/// 更改集群時/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_HashSlotMoved(object sender,HashSlotMovedEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_HashSlotMoved)}: {nameof(e.OldEndPoint)}-{e.OldEndPoint} To {nameof(e.NewEndPoint)}-{e.NewEndPoint} ");}/// <summary>/// 配置更改時/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_ConfigurationChanged(object sender, EndPointEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_ConfigurationChanged)}: {e.EndPoint}");}/// <summary>/// 發生錯誤時/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_ErrorMessage(object sender, RedisErrorEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_ErrorMessage)}: {e.Message}");}/// <summary>/// 物理連接失敗時/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_ConnectionFailed(object sender, ConnectionFailedEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_ConnectionFailed)}: {e.Exception}");}/// <summary>/// 建立物理連接時/// </summary>/// <param name="sender"></param>/// <param name="e"></param>private static void ConnMultiplexer_ConnectionRestored(object sender, ConnectionFailedEventArgs e){Console.WriteLine($"{nameof(ConnMultiplexer_ConnectionRestored)}: {e.Exception}");}#endregion#region physical save - undo#endregion} }

?3.3 引入博文a 老師的消息出列方法,代碼如下(此部分進行了部分修改)

using MVC5Project.Models; using MVC5Project.Redis.BaseOnStackExchage;namespace MVC5Project.MSMQ {public class MessageQueue{static System.Timers.Timer timer = new System.Timers.Timer(5000);public static ChatModels CurrentChatModels = new ChatModels();static MessageQueue(){timer.AutoReset = true;timer.Enabled = true;timer.Elapsed += new System.Timers.ElapsedEventHandler(timer_Elapsed);timer.Start();}private static void timer_Elapsed(object sender, System.Timers.ElapsedEventArgs e){var redisClient = new RedisHelper(2);// 消息出列CurrentChatModels = redisClient.ListLeftPop<ChatModels>("MessageQuene");}} }

3.4? web 端頁面處理,在Home的Index 頁面中

@{ } <!DOCTYPE html> <html><head><meta http-equiv="Content-Type" content="text/html; charset=utf-8" /><meta charset="utf-8" /><title>Testing MSMQ</title> </head> <body><script type="text/javascript">function onSubmit() {alert('111');$.post("/Home/Index", { userId: $('#userId').val(),chat:$('#chat').val() }, function () {});}</script> <div><form action="~/Controllers/Home" method="post"><input id="userId" value="chenk"/><input id="chat" value="this is my first messageQuene info !" /><input type="button" value="submit" οnclick="onSubmit()"/></form><label>PageInfo:</label>@ViewBag.PageInfo<label>MessageQuene:</label>@ViewBag.MessageQuene<label>Pop:</label>@ViewBag.Pop</div><script src="~/Scripts/jquery-1.10.2.min.js"></script> </body></html>

3.5 controller 處理消息入列內容

/// <summary>/// display page/// </summary>/// <returns></returns>public ActionResult Index(){var redisClient = new RedisHelper(2);ViewBag.PageInfo = "this page is Home";List<ChatModels> isError = null;ViewData["pop"] = MessageQueue.CurrentChatModels == null ? "沒有記錄" : MessageQueue.CurrentChatModels.chat;//目前ListRange()方法會出現RedisTimeOutException,并未找到問題根源,但是不影響代碼執行。ViewData["MSMQ"] = redisClient.ListRange("MessageQuene") == null ? isError = new List<ChatModels>() : redisClient.ListRange("MessageQuene").Cast<ChatModels>().ToList();return View();}

/// <summary>/// submit form action/// </summary>/// <returns></returns>[HttpPost]public ActionResult Index(FormCollection form){var redisClient = new RedisHelper(2);List<ChatModels> isError = null;//消息入列redisClient.ListRightPush("MessageQuene", new ChatModels { userId = form["userId"], chat = form["chat"] });ViewData["MessageQuene"] = redisClient.ListRange("MessageQuene") == null? isError = new List<ChatModels>(): redisClient.ListRange("MessageQuene").Cast<ChatModels>().ToList();return View();}

4、執行結果查看

a、在點擊submit 之前 db(2) 為空。

b、點擊submit

c、完成消息入列

d、5秒鐘后,數據會從db(2) 中移出。完成消息出列。

5、總結

以上,為一個完整的Redis 模擬存儲消息的實現,在過程中也有幾個未能完善的問題

(1)、StackExchage.Redis 提供的 ListRange() 方法會拋出RedisTimeOutException ,該問題并未在網上找到導致的原因。

(2)、通過ListRightPush<T>方法 插入的數據,在客戶端中并未能正確顯示,此問題還需進一步解決

總結

以上是生活随笔為你收集整理的c#进阶(4)—— Redis 用于消息队列的存储的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。