天天看點

StackExchange.Redis 之 List隊列 類型示例

//從第1個開始,依次向左插入值。如果鍵不存在,先建立再插入值  隊列形式 先進後出,後進先出
            //插入後形式   <-- 10,9,8,7,6,5,4,3,2,1 <--   方向向左依次進行
            stopwatch.Start();
            for (int i = 0; i < 10; i++)
            {
                var getobjlist = RedisCacheHelper.Instance.ListLeftPush("leftlistkey", (i + 1).ToString());
            }
            stopwatch.Stop();
            Console.WriteLine("在清單頭部插入值消耗時間:" + stopwatch.ElapsedMilliseconds.ToString());


            //從第1個開始,依次向右插入值。如果鍵不存在,先建立再插入值  先進先出,後進後出
            //插入後形式   1,2,3,4,5,6,7,8,9,10  -->   方向向右依次進行
            stopwatch.Start();
            for (int i = 0; i < 10; i++)
            {
                var getobjlist = RedisCacheHelper.Instance.ListRightPush("rightlistkey", (i + 1).ToString());
            }
            stopwatch.Stop();
            Console.WriteLine("在清單尾部插入值消耗時間:" + stopwatch.ElapsedMilliseconds.ToString());      
StackExchange.Redis 之 List隊列 類型示例
StackExchange.Redis 之 List隊列 類型示例
//擷取Left清單中的隊列元素  從清單頭部開始讀取
            var getleftvalue = RedisCacheHelper.Instance.ListRange("leftlistkey");
            Console.WriteLine(string.Join(",", getleftvalue));


            //擷取Right清單中的隊列元素   從清單頭部開始讀取
            var getrightvalue = RedisCacheHelper.Instance.ListRange("rightlistkey");
            Console.WriteLine(string.Join(",", getrightvalue));      
StackExchange.Redis 之 List隊列 類型示例
//從左邊第一個元素開始  循環移除并傳回該移除的值
            Console.WriteLine("從左邊開始");
            while (true)
            {
                var getleftvalue = RedisCacheHelper.Instance.ListLeftPop("leftlistkey");
                if (!string.IsNullOrEmpty(getleftvalue))
                {
                    Console.WriteLine("移除:" + getleftvalue);
                }
                else
                {
                    break;
                }
            }

            //從右邊第一個元素開始  循環移除并傳回該移除的值  
            Console.WriteLine("從右邊開始");
            while (true)
            {
                var getrightvalue = RedisCacheHelper.Instance.ListRightPop("rightlistkey");
                if (!string.IsNullOrEmpty(getrightvalue))
                {
                    Console.WriteLine("移除:" + getrightvalue);
                }
                else
                {
                    break;
                }
            }      
StackExchange.Redis 之 List隊列 類型示例
//從左邊第一個元素開始  循環移除并傳回該移除的值  替換一下Key後
            Console.WriteLine("從左邊開始");
            while (true)
            {
                var getleftvalue = RedisCacheHelper.Instance.ListLeftPop("rightlistkey");
                if (!string.IsNullOrEmpty(getleftvalue))
                {
                    Console.WriteLine("移除:" + getleftvalue);
                }
                else
                {
                    break;
                }
            }


            //從右邊第一個元素開始  循環移除并傳回該移除的值  替換一下Key後
            Console.WriteLine("從右邊開始");
            while (true)
            {
                var getrightvalue = RedisCacheHelper.Instance.ListRightPop("leftlistkey");
                if (!string.IsNullOrEmpty(getrightvalue))
                {
                    Console.WriteLine("移除:" + getrightvalue);
                }
                else
                {
                    break;
                }
            }      
StackExchange.Redis 之 List隊列 類型示例
//清單長度  不存在則傳回0
            var getlength = RedisCacheHelper.Instance.ListLength("leftlistkey");
            Console.WriteLine("清單長度:" + getlength);

            //删除List中的元素 并傳回删除的個數    不存在則傳回0
            var getlong = RedisCacheHelper.Instance.ListDelRange("leftlistkey", "6");
            Console.WriteLine("删除List中的元素,并傳回删除的個數:" + getlong);

            //清空清單
            RedisCacheHelper.Instance.ListClear("leftlistkey");      

使用List類型 模拟使用者并發搶購商品

//模拟資料  想List類型表中加入一定數量的庫存   50個商品
            for (int i = 1; i <= 50; i++)
            {
                var getvalue = RedisCacheHelper.Instance.ListRightPush("orderlist", i.ToString());
                //Console.WriteLine("傳回結果:" + getvalue);
            }

            //模拟建立多個使用者  100個使用者
            List<TestRedis> testlist = new List<TestRedis>();
            for (int i = 0; i < 100; i++)
            {
                testlist.Add(new TestRedis() { Uid = (i + 1) });
            }

            //先清空
            RedisCacheHelper.Instance.ListClear("orderSuccessList");

            //使用List類型模拟并發情況  不用擔心庫存為負的情況
            //模拟多個使用者搶購限時商品   100個使用者搶50個商品
            stopwatch.Start();
            List<Task> taskList = new List<Task>();
            foreach (var item in testlist)
            {
                var task = Task.Run(() =>
                {
                    try
                    {
                        //先自減,擷取自減後的值
                        long order_Num = -1;
                        long.TryParse(RedisCacheHelper.Instance.ListRightPop("orderlist"), out order_Num);
                        if (order_Num > 0)
                        {
                            //下面執行訂單邏輯(這裡不考慮業務出錯的情況)

                            RedisCacheHelper.Instance.ListLeftPush("orderSuccessList", item.Uid.ToString());  //記錄下單成功的使用者

                            //操作資料庫相關邏輯 可以使用“消息隊列”或“服務”進行資料庫同步操作

                            Console.WriteLine("使用者:" + item.Uid + ",搶到了商品:" + order_Num);
                        }
                        else
                        {
                            Console.WriteLine("商品已經被搶光了,使用者" + item.Uid + "未搶到");
                        }

                    }
                    catch (Exception ex)
                    {
                        Console.WriteLine(ex.Message);
                        throw;
                    }
                });
                taskList.Add(task);
            }
            Task.WaitAll(taskList.ToArray());
            stopwatch.Stop();
            Console.WriteLine("模拟并發場景消耗時間:" + stopwatch.ElapsedMilliseconds.ToString());      
StackExchange.Redis 之 List隊列 類型示例
StackExchange.Redis 之 List隊列 類型示例

應用場景:

Redis list 的應用場景非常多,也是 Redis 最重要的資料結構之一,比如 twitter 的關注清單,粉絲清單等都可以用 Redis 的 list 結構來實作,比較好了解,這裡不再重複。

實作方式:

Redis list 的實作為一個雙向連結清單,即可以支援反向查找和周遊,更友善操作,不過帶來了部分額外的記憶體開銷,Redis 内部的很多實作,包括發送緩沖隊列等也都是用的這個資料結構。

最後附上Helper

StackExchange.Redis 之 List隊列 類型示例
StackExchange.Redis 之 List隊列 類型示例
/// <summary>
        /// 在清單頭部插入值。如果鍵不存在,先建立再插入值
        /// </summary>
        /// <param name="redisKey"></param>
        /// <param name="redisValue"></param>
        /// <returns></returns>
        public long ListLeftPush(string redisKey, string redisValue, int db = -1)
        {
            var _db = GetDatabase(db);
            return _db.ListLeftPush(redisKey, redisValue);
        }
        /// <summary>
        /// 在清單尾部插入值。如果鍵不存在,先建立再插入值
        /// </summary>
        /// <param name="redisKey"></param>
        /// <param name="redisValue"></param>
        /// <returns></returns>
        public long ListRightPush(string redisKey, string redisValue, int db = -1)
        {
            var _db = GetDatabase(db);
            return _db.ListRightPush(redisKey, redisValue);
        }

        /// <summary>
        /// 在清單尾部插入數組集合。如果鍵不存在,先建立再插入值
        /// </summary>
        /// <param name="redisKey"></param>
        /// <param name="redisValue"></param>
        /// <returns></returns>
        public long ListRightPush(string redisKey, IEnumerable<string> redisValue, int db = -1)
        {
            var _db = GetDatabase(db);
            var redislist = new List<RedisValue>();
            foreach (var item in redisValue)
            {
                redislist.Add(item);
            }
            return _db.ListRightPush(redisKey, redislist.ToArray());
        }


        /// <summary>
        /// 移除并傳回存儲在該鍵清單的第一個元素  反序列化
        /// </summary>
        /// <param name="redisKey"></param>
        /// <returns></returns>
        public T ListLeftPop<T>(string redisKey, int db = -1) where T : class
        {
            var _db = GetDatabase(db);
            return JsonConvert.DeserializeObject<T>(_db.ListLeftPop(redisKey));
        }

        /// <summary>
        /// 移除并傳回存儲在該鍵清單的最後一個元素   反序列化
        /// 隻能是對象集合
        /// </summary>
        /// <param name="redisKey"></param>
        /// <returns></returns>
        public T ListRightPop<T>(string redisKey, int db = -1) where T : class
        {
            var _db = GetDatabase(db);
            return JsonConvert.DeserializeObject<T>(_db.ListRightPop(redisKey));
        }

        /// <summary>
        /// 移除并傳回存儲在該鍵清單的第一個元素   
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="redisKey"></param>
        /// <param name="db"></param>
        /// <returns></returns>
        public string ListLeftPop(string redisKey, int db = -1)
        {
            var _db = GetDatabase(db);
            return _db.ListLeftPop(redisKey);
        }

        /// <summary>
        /// 移除并傳回存儲在該鍵清單的最後一個元素   
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="redisKey"></param>
        /// <param name="db"></param>
        /// <returns></returns>
        public string ListRightPop(string redisKey, int db = -1)
        {
            var _db = GetDatabase(db);
            return _db.ListRightPop(redisKey);
        }

        /// <summary>
        /// 清單長度
        /// </summary>
        /// <param name="redisKey"></param>
        /// <param name="db"></param>
        /// <returns></returns>
        public long ListLength(string redisKey, int db = -1)
        {
            var _db = GetDatabase(db);
            return _db.ListLength(redisKey);
        }

        /// <summary>
        /// 傳回在該清單上鍵所對應的元素
        /// </summary>
        /// <param name="redisKey"></param>
        /// <returns></returns>
        public IEnumerable<string> ListRange(string redisKey, int db = -1)
        {
            var _db = GetDatabase(db);
            var result = _db.ListRange(redisKey);
            return result.Select(o => o.ToString());
        }

        /// <summary>
        /// 根據索引擷取指定位置資料
        /// </summary>
        /// <param name="redisKey"></param>
        /// <param name="start"></param>
        /// <param name="stop"></param>
        /// <param name="db"></param>
        /// <returns></returns>
        public IEnumerable<string> ListRange(string redisKey, int start, int stop, int db = -1)
        {
            var _db = GetDatabase(db);
            var result = _db.ListRange(redisKey, start, stop);
            return result.Select(o => o.ToString());
        }

        /// <summary>
        /// 删除List中的元素 并傳回删除的個數
        /// </summary>
        /// <param name="redisKey">key</param>
        /// <param name="redisValue">元素</param>
        /// <param name="type">大于零 : 從表頭開始向表尾搜尋,小于零 : 從表尾開始向表頭搜尋,等于零:移除表中所有與 VALUE 相等的值</param>
        /// <param name="db"></param>
        /// <returns></returns>
        public long ListDelRange(string redisKey, string redisValue, long type = 0, int db = -1)
        {
            var _db = GetDatabase(db);
            return _db.ListRemove(redisKey, redisValue, type);
        }

        /// <summary>
        /// 清空List
        /// </summary>
        /// <param name="redisKey"></param>
        /// <param name="db"></param>
        public void ListClear(string redisKey, int db = -1)
        {
            var _db = GetDatabase(db);
            _db.ListTrim(redisKey, 1, 0);
        }      

View Code

作者:PeterZhang

出處:https://www.cnblogs.com/peterzhang123

本文版權歸作者和部落格園共有,歡迎轉載,但必須給出原文連結,并保留此段聲明,否則保留追究法律責任的權利。