SpringBoot使用Redis的zset統(tǒng)計(jì)在線(xiàn)用戶(hù)信息
統(tǒng)計(jì)在線(xiàn)用戶(hù)的數(shù)量,是應(yīng)用很常見(jiàn)的需求了。如果需要精準(zhǔn)的統(tǒng)計(jì)到用戶(hù)是在線(xiàn),離線(xiàn)狀態(tài),我想只有客戶(hù)端和服務(wù)器通過(guò)保持一個(gè)TCP長(zhǎng)連接來(lái)實(shí)現(xiàn)。如果應(yīng)用本身并非一個(gè)IM應(yīng)用的話(huà),這種方式成本極高。
現(xiàn)在的應(yīng)用都趨向于使用心跳包來(lái)標(biāo)識(shí)用戶(hù)是否在線(xiàn)。用戶(hù)登錄后,每隔一段時(shí)間,往服務(wù)器推送一個(gè)消息,表示當(dāng)前用戶(hù)在線(xiàn)。服務(wù)器則可以定義一個(gè)時(shí)間差,例如:5分鐘內(nèi)收到過(guò)客戶(hù)端心跳消息,視為在線(xiàn)用戶(hù)。
在線(xiàn)用戶(hù)統(tǒng)計(jì)的實(shí)現(xiàn)
基于數(shù)據(jù)庫(kù)實(shí)現(xiàn)
最簡(jiǎn)單的辦法,就是在用戶(hù)表,添加一個(gè)最后心跳包的日期時(shí)間字段 last_active。服務(wù)器收到心跳后,每次都去更新這個(gè)字段為當(dāng)前的最新時(shí)間。
如果要查詢(xún)最近5分鐘活躍的用戶(hù)數(shù)量,就可以簡(jiǎn)單的通過(guò)一句SQL完成。
SELECT COUNT(1) AS `online_user_count` FROM `user` WHERE `last_active` BETWEEN '2020-12-22 13:00:00' AND '020-12-22 13:05:00';
弊端也是顯而易見(jiàn),為了提高檢索效率,不得不為last_active字段添加索引,而因?yàn)樾奶母?,?huì)導(dǎo)致頻繁的重新維護(hù)索引樹(shù),效率極其低下。
基于Redis實(shí)現(xiàn)
這是比較理想的一種實(shí)現(xiàn)方式了,Redis基于內(nèi)存進(jìn)行讀寫(xiě),性能自然比關(guān)系型數(shù)據(jù)庫(kù)好得多,而且它所提供的Zset可以很方便的構(gòu)建出一個(gè)在線(xiàn)用戶(hù)的統(tǒng)計(jì)服務(wù)。
Redis的Zset
這里不會(huì)涉及太多redis的東西,簡(jiǎn)單說(shuō)明以下zset。它是一個(gè)有序的set集合,集合中的每個(gè)元素由2個(gè)東西組成
- member 既然是集合,那么它便是集合中的元素,并且不能重復(fù)
- score 既然是有序的,它就是用于排序的權(quán)重字段
Zset的部分操作
添加元素
ZADD key score member [score member ...]
一次性添加一個(gè)或者多個(gè)元素到集合,如果member已經(jīng)存在則會(huì)使用當(dāng)前score進(jìn)行覆蓋
統(tǒng)計(jì)所有的元素?cái)?shù)量
ZCARD key
統(tǒng)計(jì)score值在min和max之間元素?cái)?shù)量
ZCOUNT key min max
刪除score值在min和max之間的元素
ZREMRANGEBYSCORE key min max
一個(gè)示例
我打算,用一個(gè)zset存儲(chǔ)我內(nèi)心中編程語(yǔ)言的評(píng)分排名,這個(gè)key叫做lang
添加信息,返回新添加的元素個(gè)數(shù)
> zadd lang 999 php 10 java 9 go 8 python 7 javascript "5"
查看添加的數(shù)量
> zcard lang "5"
查看評(píng)分在8 - 10之間的元素個(gè)數(shù),有3個(gè)
> zcount lang 8 10 "3"
刪除評(píng)分在8 - 1000的元素,返回刪除的個(gè)數(shù)
> ZREMRANGEBYSCORE lang 8 1000 "4"
在線(xiàn)用戶(hù)服務(wù)的實(shí)現(xiàn)
知道了zset后,就可以實(shí)現(xiàn)一個(gè)在線(xiàn)用戶(hù)的統(tǒng)計(jì)服務(wù)了。
實(shí)現(xiàn)思路
客戶(hù)端每隔5分鐘發(fā)送一個(gè)心跳到服務(wù)器,服務(wù)器根據(jù)會(huì)話(huà)獲取到用戶(hù)的ID,作為zset的member
存入zset,score便是當(dāng)前收到心跳的時(shí)間戳,當(dāng)同一個(gè)用戶(hù)第二次發(fā)送心跳的時(shí)候,就會(huì)更新他對(duì)應(yīng)的score值,由于更新是在內(nèi)存,這個(gè)速度相當(dāng)快。
zadd users 1608616915109 10000
需要統(tǒng)計(jì)出在線(xiàn)用戶(hù)的數(shù)量,本質(zhì)上就是需要統(tǒng)計(jì)出,最近5分鐘有發(fā)送心跳的用戶(hù),通過(guò)zcount可以很輕松的統(tǒng)計(jì)出來(lái)。通過(guò)程序獲取到當(dāng)前的時(shí)間戳,作為maxScore,時(shí)間戳減去5分鐘后作為minScore。
zcount users 1608616615109 1608616915109
因?yàn)槟承┯脩?hù)可能長(zhǎng)時(shí)間沒(méi)有登錄過(guò)了,可以通過(guò)ZREMRANGEBYSCORE進(jìn)行清理。通過(guò)程序獲取到當(dāng)前的時(shí)間戳,減去5分鐘后作為maxScore,使用0, 作為minScore,表示清理所有超過(guò)5分鐘沒(méi)有發(fā)送過(guò)心跳包的用戶(hù)。
ZREMRANGEBYSCORE users 0 1608616615109
實(shí)現(xiàn)代碼
import java.time.Duration;
import java.time.Instant;
import java.time.LocalDateTime;
import java.time.ZoneId;
import javax.annotation.Resource;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;
/**
*
*
* 在線(xiàn)用戶(hù)統(tǒng)計(jì)
*
* @author Administrator
*
*/
@Component
public class OnlineUserStatsService {
private static final String ONLINE_USERS = "onlie_users";
@Resource
private StringRedisTemplate stringRedisTemplate;
/**
* 添加用戶(hù)在線(xiàn)信息
* @param userId
* @return
*/
public Boolean online(Integer userId) {
return this.stringRedisTemplate.opsForZSet().add(ONLINE_USERS, userId.toString(), Instant.now().toEpochMilli());
}
/**
* 獲取一定時(shí)間內(nèi),在線(xiàn)的用戶(hù)數(shù)量
* @param duration
* @return
*/
public Long count(Duration duration) {
LocalDateTime now = LocalDateTime.now();
return this.stringRedisTemplate.opsForZSet().count(ONLINE_USERS,
now.minus(duration).atZone(ZoneId.systemDefault()).toInstant().toEpochMilli(),
now.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli());
}
/**
* 獲取所有在線(xiàn)過(guò)的用戶(hù)數(shù)量,不論時(shí)間
* @return
*/
public Long count() {
return this.stringRedisTemplate.opsForZSet().zCard(ONLINE_USERS);
}
/**
* 清除超過(guò)一定時(shí)間沒(méi)在線(xiàn)的用戶(hù)數(shù)據(jù)
* @param duration
* @return
*/
public Long clear(Duration duration) {
return this.stringRedisTemplate.opsForZSet().removeRangeByScore(ONLINE_USERS, 0,
LocalDateTime.now().minus(duration).atZone(ZoneId.systemDefault()).toInstant().toEpochMilli());
}
}
使用示例
@Resource
private OnlineUserStatsService onlineUserStatsService;
@Test
public void test() {
// ID為1的用戶(hù)發(fā)送了心跳包
boolean result = this.onlineUserStatsService.online(1);
System.out.println("online=" + result);
// 獲取5分鐘內(nèi),發(fā)送過(guò)心跳包的用戶(hù)數(shù)量,也就是在線(xiàn)用戶(hù)的數(shù)量
Long count = this.onlineUserStatsService.count(Duration.ofMinutes(5));
System.out.println("oneline count=" + count);
// 獲取所有發(fā)送過(guò)心跳包的用戶(hù)數(shù)量
count = this.onlineUserStatsService.count();
System.out.println("all count=" + count);
// 清除超過(guò)1天都沒(méi)發(fā)送過(guò)心跳包的用戶(hù)
Long clear = this.onlineUserStatsService.clear(Duration.ofDays(1));
System.out.println("clear=" + clear);
}
內(nèi)存消耗分析
可以通過(guò) http://www.redis.cn/redis_memory/ 預(yù)算Redis的內(nèi)存消耗
我對(duì)Redis的內(nèi)存分配并不熟悉,只是按照自己的想法去填寫(xiě)了一些數(shù)據(jù),所以我在這里理解的東西,可能是錯(cuò)誤的。但是我想這并不耽誤證明 - 在這種場(chǎng)景使用Zset對(duì)內(nèi)存消耗極低的事實(shí)
設(shè)想onlie_users需要存儲(chǔ)1億個(gè)用戶(hù)的狀態(tài)信息,每個(gè)元素score和member需要10個(gè)字節(jié)存儲(chǔ),那么一共大約需要20G內(nèi)存。20G的內(nèi)存對(duì)于現(xiàn)在的服務(wù)器來(lái)說(shuō),并不是大問(wèn)題。

最后
- 心跳協(xié)議不一定非要HTTP,如果客戶(hù)端支持的話(huà)UDP就很適合,可以節(jié)約一些系統(tǒng)開(kāi)銷(xiāo)。
- zset的key,不一定非要用String,可以修改序列化方式,以固定的字節(jié)的形式存儲(chǔ)用戶(hù)ID,在用戶(hù)ID過(guò)大的時(shí)候,可以節(jié)約一些存儲(chǔ)空間。
String userId = "10010"; System.out.println(userId.getBytes().length); // 以字符串形式存儲(chǔ) => 需要5個(gè)字節(jié) byte[] bin = ByteBuffer.allocate(4).putInt(Integer.valueOf(userId)).array(); System.out.println(bin.length); // 序列化為字節(jié)形式存儲(chǔ) => 需要4個(gè)字節(jié) System.out.println(ByteBuffer.wrap(bin).getInt()); // 反序列化為ID => 10010
以上就是SpringBoot使用Redis的zset統(tǒng)計(jì)在線(xiàn)用戶(hù)信息的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot統(tǒng)計(jì)在線(xiàn)用戶(hù)信息的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- springboot項(xiàng)目Redis統(tǒng)計(jì)在線(xiàn)用戶(hù)的實(shí)現(xiàn)示例
- SpringBoot+Redis Bitmap實(shí)現(xiàn)活躍用戶(hù)統(tǒng)計(jì)
- SpringBoot+Redis?BitMap實(shí)現(xiàn)簽到與統(tǒng)計(jì)的項(xiàng)目實(shí)踐
- 微服務(wù)Spring Boot 整合 Redis 實(shí)現(xiàn)UV 數(shù)據(jù)統(tǒng)計(jì)的詳細(xì)過(guò)程
- 微服務(wù)?Spring?Boot?整合?Redis?BitMap?實(shí)現(xiàn)?簽到與統(tǒng)計(jì)功能
- SpringBoot整合Redis實(shí)現(xiàn)訪(fǎng)問(wèn)量統(tǒng)計(jì)的示例代碼
- SpringBoot運(yùn)用Redis統(tǒng)計(jì)用戶(hù)在線(xiàn)數(shù)量的兩種方法實(shí)現(xiàn)
相關(guān)文章
基于eclipse.ini內(nèi)存設(shè)置的問(wèn)題詳解
本篇文章是對(duì)eclipse.ini內(nèi)存設(shè)置的問(wèn)題進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
Java實(shí)例講解多態(tài)數(shù)組的使用
本文章向大家介紹Java多態(tài)數(shù)組,主要包括Java多態(tài)數(shù)組使用實(shí)例、基本知識(shí)點(diǎn)總結(jié)和需要注意事項(xiàng),具有一定的參考價(jià)值,需要的朋友可以參考一下2022-05-05
Springboot實(shí)現(xiàn)多數(shù)據(jù)源切換詳情
這篇文章主要介紹了Springboot實(shí)現(xiàn)多數(shù)據(jù)源切換詳情,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,感興趣的朋友可以參考一下2022-09-09
SpringBoot利用MDC機(jī)制過(guò)濾單次請(qǐng)求的所有日志
在服務(wù)出現(xiàn)故障時(shí),我們經(jīng)常需要獲取一次請(qǐng)求流程里的所有日志進(jìn)行定位 ,如何將一次數(shù)據(jù)上報(bào)請(qǐng)求中包含的所有業(yè)務(wù)日志快速過(guò)濾出來(lái),就是本文要介紹的,需要的朋友可以參考下2024-04-04
ThreadPoolExecutor中的submit()方法詳細(xì)講解
在使用線(xiàn)程池的時(shí)候,發(fā)現(xiàn)除了execute()方法可以執(zhí)行任務(wù)外,還發(fā)現(xiàn)有一個(gè)方法submit()可以執(zhí)行任務(wù),本文就詳細(xì)的介紹一下ThreadPoolExecutor中的submit()方法,具有一定的參考價(jià)值,感興趣的可以了解一下2022-04-04
Mybatis-Plus自動(dòng)生成的數(shù)據(jù)庫(kù)id過(guò)長(zhǎng)的解決
這篇文章主要介紹了Mybatis-Plus自動(dòng)生成的數(shù)據(jù)庫(kù)id過(guò)長(zhǎng)的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12
詳解如何使用SpringBoot實(shí)現(xiàn)下載JSON文件
在?Spring?Boot?中實(shí)現(xiàn)文件下載功能,可以通過(guò)將?JSON?字符串作為文件內(nèi)容返回給客戶(hù)端從而實(shí)現(xiàn)JSON文件下載效果,下面我們就來(lái)看看具體操作吧2025-02-02
Java應(yīng)用啟動(dòng)停止重啟Shell腳本模板server.sh
這篇文章主要為大家介紹了Java應(yīng)用啟動(dòng)、停止、重啟Shell腳本模板server.sh,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-08-08
IDEA整合Dubbo+Zookeeper+SpringBoot實(shí)現(xiàn)
初學(xué)者,想自己動(dòng)手做一個(gè)簡(jiǎn)單的demo,本文主要介紹了IDEA整合Dubbo+Zookeeper+SpringBoot實(shí)現(xiàn),需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-06-06

