.Net Redis实战——实现文章投票并排序
本系列文章为学习Redis实战一书记录的随笔。
软件和环境版本:Redis:5.0.7 .Net 5.0
文中不会对Redis基础概念做过多介绍。
Redis数据类型和命令可在菜鸟教程学习:https://www.runoob.com/redis/redis-tutorial.html 。
示例介绍
实现一个简单的文章投票功能,并根据文章投票得分进行排序展示。
功能设计
使用散列来存储文章信息
article:92617中的冒号只是作为分隔符,可以根据个人信号替换成 | , / 等符号

使用两个有序集合来有序地存储文章:第一个有序集合的成员为文章 ID,分值为文章的发布时间;第二个有序集合的成员同样为文章 ID,而分值则为文章的评分。通过这两个有序集合,网站既可以根据文章发布的先后顺序来展示文章,又可以根据文章评分的高低来展示文章。

同时为了防止一个用户对一篇文章进行多次投票,需要为每篇文章增加一个投票用户集合存储已投票用户ID。

代码实现
代码所用Redis库为StackExchange.Redis。
具体使用参考:https://stackexchange.github.io/StackExchange.Redis/
我本机Redis安装在Windows的Linux子系统内,写数据持久化时报错“MISCONF Redis is configured to save RDB snapshots, but is currently not able to persist on disk”。
解决方案:https://blog.csdn.net/u014071875/article/details/103715183
1.初始化文章,模拟文章发布
[HttpGet("InitArticles")]
public string InitArticles()
{
using (var redis = ConnectionMultiplexer.Connect(redisConnectionStr))
{
var db = redis.GetDatabase();
foreach (var article in Articles)
{
var articleId = $"article:{article.ID}";
var hashEntiies = new [] {
new HashEntry(nameof(article.Title),article.Title),
new HashEntry(nameof(article.Link),article.Link),
new HashEntry(nameof(article.Poster),article.Poster),
new HashEntry(nameof(article.Votes),article.Votes),
new HashEntry(nameof(article.UnixTime),article.UnixTime),
};
//存储文章信息
db.HashSetAsync(articleId, hashEntiies);
//将文章信息添加到评分和时间排序的有序集合
db.SortedSetAddAsync("score:",articleId, 0);
db.SortedSetAddAsync("time:",articleId, DateTimeOffset.UtcNow.ToUnixTimeMilliseconds());
Thread.Sleep(1000);
}
}
return "文章初始化完成";
}
2.文章投票代码
[HttpGet("ArticleVote")]
public string ArticleVote(int userId,int articleId) {
using (var redis = ConnectionMultiplexer.Connect(redisConnectionStr))
{
var db = redis.GetDatabase();
if (db.SetAdd($"voted:{articleId}",$"user:{userId}"))
{
//增加投票票数和评分
db.HashIncrementAsync($"article:{articleId}", "Votes", 1);
db.SortedSetIncrementAsync("score:", $"article:{articleId}", 1);
}
else
{
return "不能重复投票";
}
}
return "投票成功";
}
3.获取文章列表(只列出根据评分排序)
[HttpGet("GetArticles")]
public IEnumerable<Article> GetArticles(int page,int size) {
var articles = new List<Article>();
using (var redis = ConnectionMultiplexer.Connect(redisConnectionStr))
{
var start = (page - 1) * size;
var end = start + size - 1;
var db = redis.GetDatabase();
var ids = db.SortedSetRangeByRank("score:", start, end,order:Order.Descending);
foreach (var id in ids)
{
var articleHashEntities = db.HashGetAll(id.ToString());
//这里除了用反射没有找到好的解决办法
var article = new Article {
ID = int.Parse(id.ToString().Split(':')[1]),
Title= articleHashEntities.FirstOrDefault(a=>a.Name=="Title").Value,
Link= articleHashEntities.FirstOrDefault(a => a.Name == "Link").Value,
Poster= int.Parse(articleHashEntities.FirstOrDefault(a => a.Name == "Poster").Value),
UnixTime= long.Parse(articleHashEntities.FirstOrDefault(a => a.Name == "UnixTime").Value),
Votes= int.Parse(articleHashEntities.FirstOrDefault(a => a.Name == "Votes").Value)
};
articles.Add(article);
}
}
return articles;
}
.Net Redis实战——实现文章投票并排序的更多相关文章
- Redis 实战 —— 02. Redis 简单实践 - 文章投票
需求 功能: P15 发布文章 获取文章 文章分组 投支持票 数值及限制条件 P15 如果一篇文章获得了至少 200 张支持票,那么这篇文章就是一篇有趣的文章 如果这个网站每天有 50 篇有趣的文章, ...
- Redis in Action 文章投票
原书用 Python 与 Redis 进行交互,我用 PHP 来实现. 环境:LNMP(CentOS 6.6 + Nginx 1.8.0 + MySQL 5.6.23 + PHP 5.6.9)+ Re ...
- 使用redis构建文章投票系统
首先,我得说明这篇博客基本上就是<<redis in action>>第一章内容的读书笔记. 需求 首先,说明一下,我们的需求 用户可以发表文章,发表时,自己就默认的给自己的文 ...
- Redis实现文章投票功能
Redis的具体操作这里就不说了,说一下需求和设计思路. 需求:自己实现一个文章投票的功能1.能够按照时间分页倒叙查看文章信息2.能够给文章投票,一个用户给一篇文章只能投票一次3.需要记录分值.每次投 ...
- 使用Redis构建文章投票网站
涉及到的key: 1. article_time, 记录文章的发布时间,zset结构 2. article_score, 记录文章的得分, zset结构 得分 = 发布时间 + 投票用户数 X 432 ...
- redis 实例2 构建文章投票网站后端
redis 实例2 构建文章投票网站后端 1.限制条件 一.如果网站获得200张支持票,那么这篇文章被设置成有趣的文章 二.如果网站发布的文章中有一定数量被认定为有趣的文章,那么这些文章需要被设置 ...
- redis实战笔记(3)-第3章 Redis命令
第3章 Redis命令 本章主要内容 字符串命令. 列表命令和集合命令 散列命令和有序集合命令 发布命令与订阅命令 其他命令 在每个不同的数据类型的章节里, 展示的都是该数据类型所独有的. 最 ...
- redis实战笔记(1)-第1章 初识Redis
第1章 初识Redis 注:本书在redis3.0版本的,比如redis3.0以后支持服务端集群.3.0之前只能客户端分片. 本章主要内容 1.Redis与其他软件的相同之处和不同之处 2.Re ...
- Redis 实战 —— 05. Redis 其他命令简介
发布与订阅 P52 Redis 实现了发布与订阅(publish/subscribe)模式,又称 pub/sub 模式(与设计模式中的观察者模式类似).订阅者负责订阅频道,发送者负责向频道发送二进制字 ...
随机推荐
- API文档大集合
[API文档大集合] sklearn API:http://sklearn.apachecn.org/cn/0.19.0/tutorial/index.html pandas API:http://p ...
- 『政善治』Postman工具 — 3、补充:restful风格接口的项目说明
目录 (一)RESTful架构风格特点 1.统一接口风格 2.规范的HTTP请求方法 3.HTTP响应码 4.什么是无状态 (二)JSON数据格式说明 1.什么是JSON 2.JSON格式的特点 3. ...
- overflow和absolute之间的问题,transfrom可以解决
CSS代码: .overflow { width: 191px; height: 191px; border: 2px solid #beceeb; overflow: hidden; } .over ...
- POJ2044 深搜+剪枝(云彩下雨)
题意: 有一个城镇,是4*4的大小的,然后你控制一块云彩,2*2的,你每天可以有9种走的方法,上下左右,或者不动,走的时候可以走1或者2步,云彩所在的地方肯定会下雨,然后给你做多365天 ...
- [CTF]思维导向图
[CTF]思维导向图 ---------------来自大佬的CTF思维导向图 Angel_Kitty https://www.cnblogs.com/ECJTUACM-873284962/ 给信息安 ...
- 使用DevExpress的GridControl实现多层级或无穷级的嵌套列表展示
在我早期的随笔<在GridControl表格控件中实现多层级主从表数据的展示>中介绍过GridControl实现二级.三级的层级列表展示,主要的逻辑就是构建GridLevelNode并添加 ...
- Spring事务明明开启了,为什么没起作用???
一.事务的特性(ACID) 1.原子性(Atomicity):事务是一个原子操作,由一系列动作组成.事务的原子性确保动作要么全部完成,要么完全不起作用. 2.一致性(Consistency):执行事务 ...
- KMP算法以及优化(代码分析以及求解next数组和nextval数组)
KMP算法以及优化(代码分析以及求解next数组和nextval数组) 来了,数据结构及算法的内容来了,这才是我们的专攻,前面写的都是开胃小菜,本篇文章,侧重考研408方向,所以保证了你只要看懂了,题 ...
- 基于pyqt5和openpyxl和Pyinstaller的青年大学习检查未学习人数的脚本
前几天接到团支书的一个需求,因为学校给的名单是青年大学习已学习的名单,然而要知道未学习的名单只能从所有团员中再排查一次,过程相当麻烦.团支书跟我抱怨后,刚好我也学过一些操作办公软件的基础.打包pyth ...
- ES6对象的新增方法的使用
Object.assign Object Object.assign(target, ...sources) 将所有可枚举属性的值从一个或多个源对象复制到目标对象 参数: target 目标对象 so ...