Java实现排行榜基于Redis
前言
排行榜作为互联网应用中几乎必不可少的一个元素,其能够勾起人类自身对比的欲望,从而来增加商品的销量。排行榜的实现方式基本大同小异,大部分都基于 Redis 的有序集合 sorted set 来实现。不久前,负责开发一个活动,就有排行榜这个需求,笔者也使用 Redis 进行了实现。本文通过了商品销售排行榜这一模型,来进行演示。
需求
- 按照商品销量进行排行
- 可以获得指定商品的排名
- 显示实时销售动态情况
需求分析
分析需求,以上这些都可以通过 Redis 的有序集合相关命令进行实现,首先看一下使用到的具体 Redis 命令。
redis> ZADD bangdan 1 "one"
(integer) 1
# 对有序集合中指定成员的分数加上增量
redis> zadd bangdan 1 "one" 4 "three" 3 "two"
(integer) 2
# 将一个或多个成员以及分数加入到有序集合中
redis> zrange bangdan 0 1
1) "one"
2) "three"
# 按照 score 升序排列 ,取出前两名
redis> zscore bangdan three
"4"
# 获得榜单中指定元素的score
redis> zrank bangdan one
(integer) 0
# 在升序榜中的名次 第一返回0
# 第三个需求需要使用 Redis 的 list 来进行实现
redis> LPUSH dynamic abc
(integer) 1
# 向队列左侧头部 push 数据
redis> LPUSH dynamic 0
"abc"
# 通过索引获取列表中的元素
redis> LTRIM dynamic 0 2
"abc"
# 对一个列表进行修剪(trim),就是说,让列表只保留指定区间内的元素,不在指定区间之内的元素都将被删除
排行榜预览
按照需求开发,最后的效果如下:

以下通过 Java 代码实现。
通过 Java 实现排行榜
引入依赖
项目中使用到了 Redis,因此需要引入相关依赖,为了简明演示,这里没有使用 JedisPool。
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.7.3</version>
</dependency>
开发逻辑
页面上点击一次购买按钮,则对该手机的销量加 1,同时将销售动态添加到队列当中
jedis.zincrby(Constants.SALES_LIST, 1, String.valueOf(phoneId));
jedis.lpush(Constants.BUY_DYNAMIC, msg);
获得排行榜
// 按照scope升序排名,取出前五
jedis.zrevrangeWithScores(Constants.SALES_LIST, 0, 4);
获得指定手机的排名情况
jedis.zrevrank(Constants.SALES_LIST, String.valueOf(phoneId));
获得销售动态,此处只取 3 条,同时队列只保存最新的 20 条动态
List<DynamicVO> dynamicList = new ArrayList<>();
for (int i = 0; i < 3; i++) {
String result = jedis.lindex(Constants.BUY_DYNAMIC, i);
if (StringUtils.isEmpty(result)) {
break;
}
String[] arr = result.split(Constants.separator);
long time = Long.valueOf(arr[0]);
String phone = arr[1];
DynamicVO vo = new DynamicVO();
vo.setPhone(phone);
vo.setTime(StringUtil.showTime(new Date(time)));
dynamicList.add(vo);
}
jedis.ltrim(Constants.BUY_DYNAMIC, 0, 19);因为排行榜这种实时性比较强的数据,更新比较快,个人觉得没有必要进行持久化,如果 Redis 的排行榜数据丢失,可以通过代码重新计算排行,通过 zadd 命令,重新添加到 Redis 中即可。
Map<String, Double> map = new HashMap<>();
map.put("1", 4.0);
map.put("2", 2.0);
map.put("3", 3.0);
jedis.zadd(Constants.SALES_LIST, map);
源码下载
Java实现排行榜基于Redis的更多相关文章
- 基于Redis位图实现用户签到功能
场景需求 适用场景如签到送积分.签到领取奖励等,大致需求如下: 签到1天送1积分,连续签到2天送2积分,3天送3积分,3天以上均送3积分等. 如果连续签到中断,则重置计数,每月初重置计数. 当月签到满 ...
- 基于redis排行榜的实战总结
前言: 之前写过排行榜的设计和实现, 不同需求其背后的架构和设计模型也不一样. 平台差异, 有的立足于游戏平台, 为多个应用提供服务, 有的仅限于单个游戏.排名范围差异, 有的面向全局排名, 有的只做 ...
- 想知道谁是你的最佳用户?基于Redis实现排行榜周期榜与最近N期榜
本文由云+社区发表 前言 业务已基于Redis实现了一个高可用的排行榜服务,长期以来相安无事.有一天,产品说:我要一个按周排名的排行榜,以反映本周内用户的活跃情况.于是周榜(按周重置更新的榜单)诞生了 ...
- Java基于redis实现分布式锁(SpringBoot)
前言 分布式锁,其实原理是就是多台机器,去争抢一个资源,谁争抢成功,那么谁就持有了这把锁,然后去执行后续的业务逻辑,执行完毕后,把锁释放掉. 可以通过多种途径实现分布式锁,例如利用数据库(mysql等 ...
- 基于redis分布式缓存实现(新浪微博案例)
第一:Redis 是什么? Redis是基于内存.可持久化的日志型.Key-Value数据库 高性能存储系统,并提供多种语言的API. 第二:出现背景 数据结构(Data Structure)需求越来 ...
- 基于redis分布式缓存实现
Redis的复制功能是完全建立在之前我们讨论过的基 于内存快照的持久化策略基础上的,也就是说无论你的持久化策略选择的是什么,只要用到了Redis的复制功能,就一定会有内存快照发生,那么首先要注意你 的 ...
- 基于redis实现的分布式锁
基于redis实现的分布式锁 我们知道,在多线程环境中,锁是实现共享资源互斥访问的重要机制,以保证任何时刻只有一个线程在访问共享资源.锁的基本原理是:用一个状态值表示锁,对锁的占用和释放通过状态值来标 ...
- 基于Redis的CAS服务端集群
为了保证生产环境CAS(Central Authentication Service)认证服务的高可用,防止出现单点故障,我们需要对CAS Server进行集群部署. CAS的Ticket默认是以Ma ...
- 基于Redis的在线用户列表解决方案
前言: 由于项目需求,需要在集群环境下实现在线用户列表的功能,并依靠在线列表实现用户单一登陆(同一账户只能一处登陆)功能: 在单机环境下,在线列表的实现方案可以采用SessionListener来完成 ...
随机推荐
- VS2008 + QGIS1.7.1试验
今天试验了一下.结果算是成功了吧.显示Generate done,生成了.但是提示了一个“SVN version不明确”的错误提示,应该无大碍吧.但是打开Build成的.sln也没看出有啥不妥. 用C ...
- 主题模型之概率潜在语义分析(Probabilistic Latent Semantic Analysis)
上一篇总结了潜在语义分析(Latent Semantic Analysis, LSA),LSA主要使用了线性代数中奇异值分解的方法,但是并没有严格的概率推导,由于文本文档的维度往往很高,如果在主题聚类 ...
- 牛客网2018暑期训练 第三场 a题
#include <bits/stdc++.h> using namespace std; vector<int> path; ; short dp[maxn][maxn][m ...
- Mysql 分区(range,list,hash)转载
MySQL支持RANGE,LIST,HASH和KEY四种分区.其中,每个分区又都有一种特殊的类型.对于RANGE分区,有RANGE COLUMNS分区.对于LIST分区,有LIST COLUMNS分区 ...
- 关于getProperties的一点记录
写了一很简单的获取配置文件的代码,结果怎么都在报空指针,经过上网查,直到要这样写才不会报: InputStream is = getClass().getClassLoader().getResour ...
- css中“~”和“>”是什么意思
p~ul选择器 p之后出现的所有ul. 两种元素必须拥有相同的父元素,但是 ul不必直接紧随 p. css中“>”是: css3特有的选择器,A>B 表示选择A元素的所有子B元素. 与A ...
- winform执行程序报错:已停止工作,windows正在检查该问题的解决方案
每次运行程序时都会弹出错误框:winform已停止工作,windows正在检查该问题的解决方案 事件查看器错位信息: 错误应用程序名称: TMS_winform.exe,版本: 1.0.0.0,时间戳 ...
- jsp的两个include了解
在jsp中三大指令(page -- include -- taglib)中有一个 include 然后再三个动作标签中(include -- forward -- param)中也有一个includ ...
- powerDesigner 把name项添加到注释(comment)
第一次写博客,分享一点经验吧,平时大家用powerDesigner的时候,pd是不会把name项默认添加到comment的,所以生成的数据库表里面也没有中文字段的注释. 我在网上查了一下.有解决方案了 ...
- linux 服务器脚本采集数据中文无法执行错误
问题描述:在 RHEL6 版本的服务器上使用脚本操作数据库,其中一个SQL的字段值为中文,每次使用 crontab定时 执行该脚本无法获取数据,实现预期效果,而手动执行正常. oracle clien ...