欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot使用Redis的zset統(tǒng)計在線用戶信息

 更新時間:2021年04月27日 09:48:47   作者:SpringBoot中文社區(qū)  
這篇文章主要介紹了SpringBoot使用Redis的zset統(tǒng)計在線用戶信息,幫助大家更好的理解和學習使用SpringBoot框架,感興趣的朋友可以了解下

統(tǒng)計在線用戶的數量,是應用很常見的需求了。如果需要精準的統(tǒng)計到用戶是在線,離線狀態(tài),我想只有客戶端和服務器通過保持一個TCP長連接來實現。如果應用本身并非一個IM應用的話,這種方式成本極高。

現在的應用都趨向于使用心跳包來標識用戶是否在線。用戶登錄后,每隔一段時間,往服務器推送一個消息,表示當前用戶在線。服務器則可以定義一個時間差,例如:5分鐘內收到過客戶端心跳消息,視為在線用戶。

在線用戶統(tǒng)計的實現

基于數據庫實現

最簡單的辦法,就是在用戶表,添加一個最后心跳包的日期時間字段 last_active。服務器收到心跳后,每次都去更新這個字段為當前的最新時間。

如果要查詢最近5分鐘活躍的用戶數量,就可以簡單的通過一句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';

弊端也是顯而易見,為了提高檢索效率,不得不為last_active字段添加索引,而因為心跳的更新,會導致頻繁的重新維護索引樹,效率極其低下。

基于Redis實現

這是比較理想的一種實現方式了,Redis基于內存進行讀寫,性能自然比關系型數據庫好得多,而且它所提供的Zset可以很方便的構建出一個在線用戶的統(tǒng)計服務。

Redis的Zset

這里不會涉及太多redis的東西,簡單說明以下zset。它是一個有序的set集合,集合中的每個元素由2個東西組成

  • member 既然是集合,那么它便是集合中的元素,并且不能重復
  • score  既然是有序的,它就是用于排序的權重字段

Zset的部分操作

添加元素

ZADD key score member [score member ...]

一次性添加一個或者多個元素到集合,如果member已經存在則會使用當前score進行覆蓋

統(tǒng)計所有的元素數量

ZCARD key

統(tǒng)計score值在min和max之間元素數量

ZCOUNT key min max

刪除score值在min和max之間的元素

ZREMRANGEBYSCORE key min max

一個示例

我打算,用一個zset存儲我內心中編程語言的評分排名,這個key叫做lang

添加信息,返回新添加的元素個數

> zadd lang 999 php 10 java 9 go 8 python 7 javascript
"5"

查看添加的數量

> zcard lang
"5"

查看評分在8 - 10之間的元素個數,有3個

> zcount lang 8 10
"3"

刪除評分在8 - 1000的元素,返回刪除的個數

> ZREMRANGEBYSCORE lang 8 1000
"4"

在線用戶服務的實現

知道了zset后,就可以實現一個在線用戶的統(tǒng)計服務了。

實現思路

客戶端每隔5分鐘發(fā)送一個心跳到服務器,服務器根據會話獲取到用戶的ID,作為zset的member
存入zset,score便是當前收到心跳的時間戳,當同一個用戶第二次發(fā)送心跳的時候,就會更新他對應的score值,由于更新是在內存,這個速度相當快。

zadd users 1608616915109 10000

需要統(tǒng)計出在線用戶的數量,本質上就是需要統(tǒng)計出,最近5分鐘有發(fā)送心跳的用戶,通過zcount可以很輕松的統(tǒng)計出來。通過程序獲取到當前的時間戳,作為maxScore,時間戳減去5分鐘后作為minScore。

zcount users 1608616615109 1608616915109 

因為某些用戶可能長時間沒有登錄過了,可以通過ZREMRANGEBYSCORE進行清理。通過程序獲取到當前的時間戳,減去5分鐘后作為maxScore,使用0, 作為minScore,表示清理所有超過5分鐘沒有發(fā)送過心跳包的用戶。

ZREMRANGEBYSCORE users 0 1608616615109 

實現代碼

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;

/**
 * 
 * 
 * 在線用戶統(tǒng)計
 * 
 * @author Administrator
 *
 */
@Component
public class OnlineUserStatsService {
    
    private static final String ONLINE_USERS = "onlie_users";

    @Resource
    private StringRedisTemplate stringRedisTemplate;

    /**
     * 添加用戶在線信息
     * @param userId
     * @return 
     */
    public Boolean online(Integer userId) {
        return this.stringRedisTemplate.opsForZSet().add(ONLINE_USERS, userId.toString(), Instant.now().toEpochMilli());
    }
    
    /**
     * 獲取一定時間內,在線的用戶數量
     * @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());
    }
    
    /**
     * 獲取所有在線過的用戶數量,不論時間
     * @return
     */
    public Long count() {
        return this.stringRedisTemplate.opsForZSet().zCard(ONLINE_USERS);
    }
    
    /**
     * 清除超過一定時間沒在線的用戶數據
     * @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的用戶發(fā)送了心跳包
    boolean result = this.onlineUserStatsService.online(1);
    System.out.println("online=" + result);
    
    // 獲取5分鐘內,發(fā)送過心跳包的用戶數量,也就是在線用戶的數量
    Long count = this.onlineUserStatsService.count(Duration.ofMinutes(5));
    System.out.println("oneline count=" + count);
    
    // 獲取所有發(fā)送過心跳包的用戶數量
    count = this.onlineUserStatsService.count();
    System.out.println("all count=" + count);
    
    // 清除超過1天都沒發(fā)送過心跳包的用戶
    Long clear = this.onlineUserStatsService.clear(Duration.ofDays(1));
    System.out.println("clear=" + clear);
}

內存消耗分析

可以通過 http://www.redis.cn/redis_memory/ 預算Redis的內存消耗

我對Redis的內存分配并不熟悉,只是按照自己的想法去填寫了一些數據,所以我在這里理解的東西,可能是錯誤的。但是我想這并不耽誤證明 - 在這種場景使用Zset對內存消耗極低的事實

設想onlie_users需要存儲1億個用戶的狀態(tài)信息,每個元素score和member需要10個字節(jié)存儲,那么一共大約需要20G內存。20G的內存對于現在的服務器來說,并不是大問題。

最后

  • 心跳協(xié)議不一定非要HTTP,如果客戶端支持的話UDP就很適合,可以節(jié)約一些系統(tǒng)開銷。
  • zset的key,不一定非要用String,可以修改序列化方式,以固定的字節(jié)的形式存儲用戶ID,在用戶ID過大的時候,可以節(jié)約一些存儲空間。
String userId = "10010";
System.out.println(userId.getBytes().length); // 以字符串形式存儲 => 需要5個字節(jié)

byte[] bin = ByteBuffer.allocate(4).putInt(Integer.valueOf(userId)).array();
System.out.println(bin.length);                    // 序列化為字節(jié)形式存儲 => 需要4個字節(jié)

System.out.println(ByteBuffer.wrap(bin).getInt());    // 反序列化為ID => 10010

以上就是SpringBoot使用Redis的zset統(tǒng)計在線用戶信息的詳細內容,更多關于SpringBoot統(tǒng)計在線用戶信息的資料請關注腳本之家其它相關文章!

相關文章

  • 基于eclipse.ini內存設置的問題詳解

    基于eclipse.ini內存設置的問題詳解

    本篇文章是對eclipse.ini內存設置的問題進行了詳細的分析介紹,需要的朋友參考下
    2013-05-05
  • Java實例講解多態(tài)數組的使用

    Java實例講解多態(tài)數組的使用

    本文章向大家介紹Java多態(tài)數組,主要包括Java多態(tài)數組使用實例、基本知識點總結和需要注意事項,具有一定的參考價值,需要的朋友可以參考一下
    2022-05-05
  • Springboot實現多數據源切換詳情

    Springboot實現多數據源切換詳情

    這篇文章主要介紹了Springboot實現多數據源切換詳情,文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,感興趣的朋友可以參考一下
    2022-09-09
  • SpringBoot利用MDC機制過濾單次請求的所有日志

    SpringBoot利用MDC機制過濾單次請求的所有日志

    在服務出現故障時,我們經常需要獲取一次請求流程里的所有日志進行定位 ,如何將一次數據上報請求中包含的所有業(yè)務日志快速過濾出來,就是本文要介紹的,需要的朋友可以參考下
    2024-04-04
  • ThreadPoolExecutor中的submit()方法詳細講解

    ThreadPoolExecutor中的submit()方法詳細講解

    在使用線程池的時候,發(fā)現除了execute()方法可以執(zhí)行任務外,還發(fā)現有一個方法submit()可以執(zhí)行任務,本文就詳細的介紹一下ThreadPoolExecutor中的submit()方法,具有一定的參考價值,感興趣的可以了解一下
    2022-04-04
  • SpringMVC中的HandlerAdapter解析

    SpringMVC中的HandlerAdapter解析

    這篇文章主要介紹了SpringMVC中的HandlerAdapter解析,HandlerAdapter是一個關鍵的組件,用于將請求與處理程序方法進行適配和調度,它充當了控制器和處理程序之間的橋梁,負責將請求的參數和處理程序方法進行匹配,并將結果返回給前端,需要的朋友可以參考下
    2023-10-10
  • Mybatis-Plus自動生成的數據庫id過長的解決

    Mybatis-Plus自動生成的數據庫id過長的解決

    這篇文章主要介紹了Mybatis-Plus自動生成的數據庫id過長的解決,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • 詳解如何使用SpringBoot實現下載JSON文件

    詳解如何使用SpringBoot實現下載JSON文件

    在?Spring?Boot?中實現文件下載功能,可以通過將?JSON?字符串作為文件內容返回給客戶端從而實現JSON文件下載效果,下面我們就來看看具體操作吧
    2025-02-02
  • Java應用啟動停止重啟Shell腳本模板server.sh

    Java應用啟動停止重啟Shell腳本模板server.sh

    這篇文章主要為大家介紹了Java應用啟動、停止、重啟Shell腳本模板server.sh,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-08-08
  • IDEA整合Dubbo+Zookeeper+SpringBoot實現

    IDEA整合Dubbo+Zookeeper+SpringBoot實現

    初學者,想自己動手做一個簡單的demo,本文主要介紹了IDEA整合Dubbo+Zookeeper+SpringBoot實現,需要的朋友們下面隨著小編來一起學習學習吧
    2021-06-06

最新評論