Redis如何使用zset處理排行榜和計(jì)數(shù)問(wèn)題
Redis使用zset處理排行榜和計(jì)數(shù)
在處理計(jì)數(shù)業(yè)務(wù)時(shí),我們一般會(huì)使用一個(gè)數(shù)據(jù)結(jié)構(gòu),既是集合又可以保證唯一性,所以我們會(huì)選擇Redis中的set集合:
業(yè)務(wù)邏輯
用戶點(diǎn)擊點(diǎn)贊按鈕,需要再set集合內(nèi)判斷是否已點(diǎn)贊,未點(diǎn)贊則需要將點(diǎn)贊數(shù)+1并保存用戶信息到集合中,已點(diǎn)贊則需要將數(shù)據(jù)庫(kù)點(diǎn)贊數(shù)-1并移除set集合中的用戶。
@Service
public class BlogServiceImpl extends ServiceImpl<BlogMapper, Blog> implements IBlogService {
@Autowired
private IUserService userService;
@Resource
private StringRedisTemplate stringRedisTemplate;
@Override
public Result likeBlog(Long id) {
// 獲取登錄用戶
Long userId = UserHolder.getUser().getId();
// 判斷當(dāng)前登錄用戶是否已經(jīng)點(diǎn)贊
String key = "blog:like:" + id;
Boolean isMember = stringRedisTemplate.opsForSet().isMember(key, userId.toString());
if(BooleanUtil.isFalse(isMember)){
// 未點(diǎn)贊
// 數(shù)據(jù)庫(kù)點(diǎn)贊數(shù)+1
boolean isSuccess = update().setSql("like = like + 1").eq("id",id).update();
// 保存用戶到Redis集合中
if(isSuccess){
stringRedisTemplate.opsForSet().add(key, userId.toString());
}
} else {
// 已點(diǎn)贊,取消點(diǎn)贊
// 數(shù)據(jù)庫(kù)點(diǎn)贊數(shù)-1
boolean isSuccess = update().setSql("like = like - 1").eq("id",id).update();
// 移除set集合中的用戶
stringRedisTemplate.opsForSet().remove(key, userId.toString());
}
return Result.ok();
}
}那么我們想要實(shí)現(xiàn)按照點(diǎn)贊時(shí)間的先后順序排序,返回Top5的用戶,這個(gè)時(shí)候set無(wú)法保證數(shù)據(jù)有序,所以我們需要換一個(gè)數(shù)據(jù)結(jié)構(gòu)滿足業(yè)務(wù)需求:

Redis 的 ZSET(有序集合) 是一個(gè)非常適合用于處理 排行榜 和 計(jì)數(shù)問(wèn)題 的數(shù)據(jù)結(jié)構(gòu)。
在高并發(fā)的點(diǎn)贊業(yè)務(wù)中,使用 ZSET 可以幫助我們高效地管理點(diǎn)贊的排名,并且由于 ZSET 的排序特性,我們可以輕松實(shí)現(xiàn)根據(jù)點(diǎn)贊數(shù)實(shí)時(shí)排序的功能。
ZSET 數(shù)據(jù)結(jié)構(gòu)
Redis 的 ZSET 是一個(gè)集合,它的每個(gè)元素都會(huì)關(guān)聯(lián)一個(gè) 分?jǐn)?shù)(score),這個(gè)分?jǐn)?shù)決定了元素在集合中的排序。ZSET 保證集合中的元素是按分?jǐn)?shù)排序的,并且可以在 O(log(N)) 的時(shí)間復(fù)雜度內(nèi)進(jìn)行添加、刪除和查找操作。
在高并發(fā)的點(diǎn)贊業(yè)務(wù)中,ZSET 可以幫助我們輕松地進(jìn)行以下幾項(xiàng)操作:
- 記錄每個(gè)用戶對(duì)某個(gè)內(nèi)容(如文章、評(píng)論等)的點(diǎn)贊數(shù)。
- 通過(guò)分?jǐn)?shù)進(jìn)行實(shí)時(shí)排序,獲取點(diǎn)贊數(shù)最多的內(nèi)容。
優(yōu)化高并發(fā)的點(diǎn)贊操作
在高并發(fā)情況下,當(dāng)多個(gè)用戶同時(shí)對(duì)某個(gè)內(nèi)容進(jìn)行點(diǎn)贊時(shí),我們需要高效地更新該內(nèi)容的點(diǎn)贊數(shù),并保證數(shù)據(jù)一致性。ZSET 提供了很好的支持,具體步驟如下:
- 用戶點(diǎn)贊操作:使用
ZINCRBY命令來(lái)對(duì)某個(gè)元素的分?jǐn)?shù)進(jìn)行增量操作,表示對(duì)該內(nèi)容的點(diǎn)贊數(shù)增加。 - 查看點(diǎn)贊數(shù):可以通過(guò)
ZSCORE命令獲取某個(gè)內(nèi)容的當(dāng)前點(diǎn)贊數(shù)。 - 查看排行榜:使用
ZRANGE或ZREVRANGE命令來(lái)獲取點(diǎn)贊數(shù)排名前 N 的內(nèi)容,按分?jǐn)?shù)進(jìn)行排序。
ZSET 結(jié)構(gòu)設(shè)計(jì)
key:表示某個(gè)內(nèi)容的點(diǎn)贊的 id。value:表示點(diǎn)贊用戶的 id。score:根據(jù)點(diǎn)贊時(shí)間排序。
下面是修改后的點(diǎn)贊邏輯:
@Service
public class BlogServiceImpl extends ServiceImpl<BlogMapper, Blog> implements IBlogService {
@Autowired
private IUserService userService;
@Resource
private StringRedisTemplate stringRedisTemplate;
@Override
public Result likeBlog(Long id) {
// 獲取登錄用戶
Long userId = UserHolder.getUser().getId();
// 判斷當(dāng)前登錄用戶是否已經(jīng)點(diǎn)贊
String key = "blog:like:" + id;
Double score = stringRedisTemplate.opsForZSet().score(key, userId.toString());
if(score == null){
// 未點(diǎn)贊
// 數(shù)據(jù)庫(kù)點(diǎn)贊數(shù)+1
boolean isSuccess = update().setSql("like = like + 1").eq("id",id).update();
// 保存用戶到Redis集合中
if(isSuccess){
stringRedisTemplate.opsForZSet().add(key, userId.toString(), System.currentTimeMillis());
}
} else {
// 已點(diǎn)贊,取消點(diǎn)贊
// 數(shù)據(jù)庫(kù)點(diǎn)贊數(shù)-1
boolean isSuccess = update().setSql("like = like - 1").eq("id",id).update();
// 移除set集合中的用戶
stringRedisTemplate.opsForZSet().remove(key, userId.toString());
}
return Result.ok();
}
}而點(diǎn)贊排行榜代碼如下:
@Service
public class BlogServiceImpl extends ServiceImpl<BlogMapper, Blog> implements IBlogService {
@Autowired
private IUserService userService;
@Resource
private StringRedisTemplate stringRedisTemplate;
@Override
public Result queryBlogLikes(Long id) {
String key = "blog:like:" + id;
// 查詢top5的點(diǎn)贊用戶 zrange key 0 4
Set<String> top5 = stringRedisTemplate.opsForZSet().range(key, 0, 4);
if (top5 == null || top5.isEmpty()) {
return Result.ok(Collections.emptyList());
}
// 解析出集合中的用戶的id
List<Long> ids = top5.stream().map(Long::valueOf).collect(Collectors.toList());
// 根據(jù)id查詢用戶,并將類型由User轉(zhuǎn)為UserDTO,隨后轉(zhuǎn)換為L(zhǎng)ist集合
String idStr = StrUtil.join(",",ids);
// List<UserDTO> userDTOs = userService.listByIds(ids).stream()
// .map(user -> BeanUtil.copyProperties(user, UserDTO.class))
// .collect(Collectors.toList());
List<UserDTO> userDTOs = userService.query()
.in("id",ids).last("order by field(id," + idStr +")").list()
.stream()
.map(user -> BeanUtil.copyProperties(user, UserDTO.class))
.collect(Collectors.toList());
return Result.ok(userDTOs);
}
}使用
userService.query().in("id", ids).last("order by field(id," + idStr + ")") 來(lái)查詢用戶信息,并且使用 order by field(id, ...) 語(yǔ)句來(lái)保證查詢結(jié)果的順序與 top5 中的用戶順序一致。
這里的 order by field(id, ...) 是關(guān)鍵,它確保了從數(shù)據(jù)庫(kù)返回的數(shù)據(jù)順序和 Redis 返回的 top5 用戶順序完全匹配。因?yàn)?Redis 中的 ZSet 是有順序的,top5 會(huì)按照點(diǎn)贊數(shù)量進(jìn)行排序。
如果直接使用 listByIds 方法,可能會(huì)導(dǎo)致結(jié)果順序不一致。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Redis刪除某個(gè)目錄下的數(shù)據(jù)的實(shí)現(xiàn)
本文介紹了如何在Redis中刪除指定目錄下的數(shù)據(jù),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-09-09
redis?手機(jī)驗(yàn)證碼實(shí)現(xiàn)示例
本文主要介紹了redis?手機(jī)驗(yàn)證碼實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11
spring?boot整合redis中間件與熱部署實(shí)現(xiàn)代碼
spring?boot整合redis最常用的有三個(gè)工具庫(kù)Jedis,Redisson,Lettuce,本文重點(diǎn)介紹spring?boot整合redis中間件與熱部署實(shí)現(xiàn),需要的朋友可以參考下2023-01-01
如何解決Redis緩存穿透(緩存空對(duì)象、布隆過(guò)濾器)
緩存穿透是一個(gè)常見(jiàn)的問(wèn)題,它發(fā)生當(dāng)請(qǐng)求的數(shù)據(jù)既不在緩存中也不在數(shù)據(jù)庫(kù)中,文章通過(guò)一個(gè)查詢商品店鋪的案例,展示了如何結(jié)合這兩種方法來(lái)避免緩存穿透,首先利用布隆過(guò)濾器過(guò)濾掉不存在的id,對(duì)于誤判的情況,則采用緩存空對(duì)象的策略進(jìn)行補(bǔ)救2024-11-11
Redis高并發(fā)場(chǎng)景下秒殺超賣解決方案(秒殺場(chǎng)景)
早起的12306購(gòu)票,剛被開(kāi)發(fā)出來(lái)使用的時(shí)候,12306會(huì)經(jīng)常出現(xiàn)超賣 這種現(xiàn)象,也就是說(shuō)車票只剩10張了,卻被20個(gè)人買到了,這種現(xiàn)象就是超賣,今天通過(guò)本文給大家介紹Redis高并發(fā)場(chǎng)景下秒殺超賣解決方案,感興趣的朋友一起看看吧2022-04-04
Redis中Zset類型常用命令的實(shí)現(xiàn)
Zset是Redis的一種有序集合數(shù)據(jù)類型,Zset通過(guò)壓縮列表和跳躍表兩種底層編碼方式支持小數(shù)據(jù)集和大數(shù)據(jù)集,支持多種操作,包括添加、查詢、刪除元素以及集合運(yùn)算等,具有不同的時(shí)間復(fù)雜度,感興趣的可以了解一下2024-10-10

