//從第1個開始,,依次向左插入值,。如果鍵不存在,,先創(chuàng)建再插入值 隊列形式 先進(jìn)后出,,后進(jìn)先出
//獲取Left列表中的隊列元素 從列表頭部開始讀取 var getleftvalue = RedisCacheHelper.Instance.ListRange("leftlistkey"); Console.WriteLine(string.Join(",", getleftvalue)); //獲取Right列表中的隊列元素 從列表頭部開始讀取 var getrightvalue = RedisCacheHelper.Instance.ListRange("rightlistkey"); Console.WriteLine(string.Join(",", getrightvalue)); //從左邊第一個元素開始 循環(huán)移除并返回該移除的值 Console.WriteLine("從左邊開始"); while (true) { var getleftvalue = RedisCacheHelper.Instance.ListLeftPop("leftlistkey"); if (!string.IsNullOrEmpty(getleftvalue)) { Console.WriteLine("移除:" + getleftvalue); } else { break; } } //從右邊第一個元素開始 循環(huán)移除并返回該移除的值 Console.WriteLine("從右邊開始"); while (true) { var getrightvalue = RedisCacheHelper.Instance.ListRightPop("rightlistkey"); if (!string.IsNullOrEmpty(getrightvalue)) { Console.WriteLine("移除:" + getrightvalue); } else { break; } } //從左邊第一個元素開始 循環(huán)移除并返回該移除的值 替換一下Key后 Console.WriteLine("從左邊開始"); while (true) { var getleftvalue = RedisCacheHelper.Instance.ListLeftPop("rightlistkey"); if (!string.IsNullOrEmpty(getleftvalue)) { Console.WriteLine("移除:" + getleftvalue); } else { break; } } //從右邊第一個元素開始 循環(huán)移除并返回該移除的值 替換一下Key后 Console.WriteLine("從右邊開始"); while (true) { var getrightvalue = RedisCacheHelper.Instance.ListRightPop("leftlistkey"); if (!string.IsNullOrEmpty(getrightvalue)) { Console.WriteLine("移除:" + getrightvalue); } else { break; } } //列表長度 不存在則返回0 var getlength = RedisCacheHelper.Instance.ListLength("leftlistkey"); Console.WriteLine("列表長度:" + getlength); //刪除List中的元素 并返回刪除的個數(shù) 不存在則返回0 var getlong = RedisCacheHelper.Instance.ListDelRange("leftlistkey", "6"); Console.WriteLine("刪除List中的元素,,并返回刪除的個數(shù):" + getlong); //清空列表 RedisCacheHelper.Instance.ListClear("leftlistkey"); 使用List類型 模擬用戶并發(fā)搶購商品 //模擬數(shù)據(jù) 想List類型表中加入一定數(shù)量的庫存 50個商品 for (int i = 1; i <= 50; i++) { var getvalue = RedisCacheHelper.Instance.ListRightPush("orderlist", i.ToString()); //Console.WriteLine("返回結(jié)果:" + getvalue); } //模擬創(chuàng)建多個用戶 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類型模擬并發(fā)情況 不用擔(dān)心庫存為負(fù)的情況 //模擬多個用戶搶購限時商品 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) { //下面執(zhí)行訂單邏輯(這里不考慮業(yè)務(wù)出錯的情況) RedisCacheHelper.Instance.ListLeftPush("orderSuccessList", item.Uid.ToString()); //記錄下單成功的用戶 //操作數(shù)據(jù)庫相關(guān)邏輯 可以使用“消息隊列”或“服務(wù)”進(jìn)行數(shù)據(jù)庫同步操作 Console.WriteLine("用戶:" + item.Uid + ",搶到了商品:" + order_Num); } else { Console.WriteLine("商品已經(jīng)被搶光了,用戶" + item.Uid + "未搶到"); } } catch (Exception ex) { Console.WriteLine(ex.Message); throw; } }); taskList.Add(task); } Task.WaitAll(taskList.ToArray()); stopwatch.Stop(); Console.WriteLine("模擬并發(fā)場景消耗時間:" + stopwatch.ElapsedMilliseconds.ToString());
應(yīng)用場景: Redis list 的應(yīng)用場景非常多,,也是 Redis 最重要的數(shù)據(jù)結(jié)構(gòu)之一,比如 twitter 的關(guān)注列表,,粉絲列表等都可以用 Redis 的 list 結(jié)構(gòu)來實現(xiàn),,比較好理解,這里不再重復(fù),。 實現(xiàn)方式: Redis list 的實現(xiàn)為一個雙向鏈表,,即可以支持反向查找和遍歷,更方便操作,,不過帶來了部分額外的內(nèi)存開銷,,Redis 內(nèi)部的很多實現(xiàn),包括發(fā)送緩沖隊列等也都是用的這個數(shù)據(jù)結(jié)構(gòu),。
最后附上Helper /// <summary> /// 在列表頭部插入值,。如果鍵不存在,先創(chuàng)建再插入值 /// </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> /// 在列表尾部插入值,。如果鍵不存在,,先創(chuàng)建再插入值 /// </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> /// 在列表尾部插入數(shù)組集合。如果鍵不存在,先創(chuàng)建再插入值 /// </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> /// 返回在該列表上鍵所對應(yīng)的元素 /// </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> /// 根據(jù)索引獲取指定位置數(shù)據(jù) /// </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中的元素 并返回刪除的個數(shù) /// </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); }
|
|