( i = ; i < ; i++ getobjlist = RedisCacheHelper.Instance.ListLeftPush(, (i + + ( i = ; i < ; i++ getobjlist = RedisCacheHelper.Instance.ListRightPush(, (i + + stopwatch.ElapsedMilliseconds.ToString());
//获取Left列表中的队列元素 从列表头部开始读取 var getleftvalue = RedisCacheHelper.Instance.ListRange("leftlistkey"); Console.WriteLine(string.Join(",", getleftvalue)); //获取Right列表中的队列元素 从列表头部开始读取 var getrightvalue = RedisCacheHelper.Instance.ListRange("rightlistkey"); Console.WriteLine(string.Join(",", getrightvalue));
//从左边第一个元素开始 循环移除并返回该移除的值 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; } }
//从左边第一个元素开始 循环移除并返回该移除的值 替换一下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; } }
//列表长度 不存在则返回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个用户 Listtestlist = new List(); for (int i = 0; i < 100; i++) { testlist.Add(new TestRedis() { Uid = (i + 1) }); } //先清空 RedisCacheHelper.Instance.ListClear("orderSuccessList"); //使用List类型模拟并发情况 不用担心库存为负的情况 //模拟多个用户抢购限时商品 100个用户抢50个商品 stopwatch.Start(); ListtaskList = new List(); 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());
应用场景:
Redis list 的应用场景非常多,也是 Redis 最重要的数据结构之一,比如 twitter 的关注列表,粉丝列表等都可以用 Redis 的 list 结构来实现,比较好理解,这里不再重复。
实现方式:
Redis list 的实现为一个双向链表,即可以支持反向查找和遍历,更方便操作,不过带来了部分额外的内存开销,Redis 内部的很多实现,包括发送缓冲队列等也都是用的这个数据结构。
最后附上Helper
/// /// 在列表头部插入值。如果键不存在,先创建再插入值 /// /// /// /// public long ListLeftPush(string redisKey, string redisValue, int db = -1) { var _db = GetDatabase(db); return _db.ListLeftPush(redisKey, redisValue); } /// /// 在列表尾部插入值。如果键不存在,先创建再插入值 /// /// /// /// public long ListRightPush(string redisKey, string redisValue, int db = -1) { var _db = GetDatabase(db); return _db.ListRightPush(redisKey, redisValue); } /// /// 在列表尾部插入数组集合。如果键不存在,先创建再插入值 /// /// /// /// public long ListRightPush(string redisKey, IEnumerable<string> redisValue, int db = -1) { var _db = GetDatabase(db); var redislist = new List(); foreach (var item in redisValue) { redislist.Add(item); } return _db.ListRightPush(redisKey, redislist.ToArray()); } /// /// 移除并返回存储在该键列表的第一个元素 反序列化 /// /// /// public T ListLeftPop(string redisKey, int db = -1) where T : class { var _db = GetDatabase(db); return JsonConvert.DeserializeObject(_db.ListLeftPop(redisKey)); } /// /// 移除并返回存储在该键列表的最后一个元素 反序列化 /// 只能是对象集合 /// /// /// public T ListRightPop(string redisKey, int db = -1) where T : class { var _db = GetDatabase(db); return JsonConvert.DeserializeObject(_db.ListRightPop(redisKey)); } /// /// 移除并返回存储在该键列表的第一个元素 /// /// /// /// /// public string ListLeftPop(string redisKey, int db = -1) { var _db = GetDatabase(db); return _db.ListLeftPop(redisKey); } /// /// 移除并返回存储在该键列表的最后一个元素 /// /// /// /// /// public string ListRightPop(string redisKey, int db = -1) { var _db = GetDatabase(db); return _db.ListRightPop(redisKey); } /// /// 列表长度 /// /// /// /// public long ListLength(string redisKey, int db = -1) { var _db = GetDatabase(db); return _db.ListLength(redisKey); } /// /// 返回在该列表上键所对应的元素 /// /// /// public IEnumerable<string> ListRange(string redisKey, int db = -1) { var _db = GetDatabase(db); var result = _db.ListRange(redisKey); return result.Select(o => o.ToString()); } /// /// 根据索引获取指定位置数据 /// /// /// /// /// /// 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()); } /// /// 删除List中的元素 并返回删除的个数 /// /// key /// 元素 /// 大于零 : 从表头开始向表尾搜索,小于零 : 从表尾开始向表头搜索,等于零:移除表中所有与 VALUE 相等的值 /// /// public long ListDelRange(string redisKey, string redisValue, long type = 0, int db = -1) { var _db = GetDatabase(db); return _db.ListRemove(redisKey, redisValue, type); } /// /// 清空List /// /// /// public void ListClear(string redisKey, int db = -1) { var _db = GetDatabase(db); _db.ListTrim(redisKey, 1, 0); }
View Code