亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Redis應(yīng)用之簽到的使用

 更新時間:2024年05月09日 08:30:42   作者:沉河不浮  
在很多時候,我們遇到用戶簽到的場景,本文主要介紹了Redis應(yīng)用之簽到的使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧

位圖由一系列二進制位組成,每個位可以被設(shè)置為1或0,當我們在處理需要高效存儲和操作大量二進制位數(shù)據(jù)的適合,位圖是一個非常有用的工具。
位圖操作命令有:

  • SETBIT:設(shè)置位圖中指定位置的位的值??梢詫⑽辉O(shè)置為 0 或 1。
  • GETBIT:獲取位圖中指定位置的位的值。
  • BITCOUNT:計算位圖中置為 1 的位的數(shù)量。
  • BITOP:對多個位圖執(zhí)行邏輯運算(AND、OR、XOR、NOT)。
  • BITFIELD:執(zhí)行復雜的位字段操作,允許你在位圖上進行位級別的讀寫操作。

其中,用的最多的是前三個操作,示例如下:

image.png

位圖的應(yīng)用十分廣泛,包括但不限于以下幾方面:

  • 統(tǒng)計用戶活躍度:可以使用位圖追蹤用戶的登錄活動,每個用戶對應(yīng)一個位圖,每天的登錄狀態(tài)可以用一個二進制位表示,通過 BITOP 命令可以計算多個用戶的交集,從而得到活躍用戶的統(tǒng)計信息。
  • 數(shù)據(jù)壓縮:位圖可以高效地存儲大量的二進制數(shù)據(jù),比如布隆過濾器(Bloom Filter)就是基于位圖實現(xiàn)的一種數(shù)據(jù)結(jié)構(gòu),用于快速判斷元素是否存在。
  • 事件計數(shù):可以使用位圖記錄每天不同時間段的事件發(fā)生情況,比如網(wǎng)站的訪問量,每個時間段對應(yīng)一個位圖,每次事件發(fā)生時將對應(yīng)的位設(shè)置為 1,通過 BITCOUNT 命令可以計算出每個時間段的事件數(shù)量。
  • 權(quán)限管理:可以使用位圖來管理用戶的權(quán)限,每個用戶對應(yīng)一個位圖,每個權(quán)限對應(yīng)一個二進制位,通過 BITOP 命令可以進行權(quán)限的并集、交集等操作。

RedisTemplate操作位圖

在之前的幾篇文章中,我們總結(jié)了一個Redis工具類,但是那個工具類中,并沒有和位圖相關(guān)的操作,這里添加和位圖操作相關(guān)的方法:

   // value: true為1, false為0
    public boolean setBit(String key, int offset, boolean value) {
        return redisTemplate.opsForValue().setBit(key, offset, value);
    }

    public boolean getBit(String key, int offset) {
        return redisTemplate.opsForValue().getBit(key, offset);
    }

    /**
     * 統(tǒng)計對應(yīng)值為1 的數(shù)量
     * @param key
     * @return
     */
    public long bitCount(String key) {
        if (StringUtils.isEmpty(key)) {
            return 0L;
        }
        return redisTemplate.execute((RedisCallback<Long>) con -> con.bitCount(key.getBytes()));
    }

    /**
     * 統(tǒng)計在字節(jié)范圍內(nèi),對應(yīng)值為1的數(shù)量
     * @param key
     * @param start
     * @param end
     * @return
     */
    public Long bitCount(String key, long start, long end) {
        return redisTemplate.execute((RedisCallback<Long>) con -> con.bitCount(key.getBytes(), start, end));
    }

添加測試類,用于測試位圖操作:

package org.example;

import org.example.util.RedisUtils;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

@SpringBootTest
public class RedisBitMapTest {
    @Autowired
    private RedisUtils redisUtils;

    @Test
    public void testBitMap() {
        redisUtils.setBit("bit", 0, true);
        redisUtils.setBit("bit", 1, true);
        redisUtils.setBit("bit", 3, true);
        redisUtils.setBit("bit", 7, true);
        System.out.println(redisUtils.bitCount("bit"));

    }
}

執(zhí)行結(jié)果如下:

image.png

我們通過Redis可視化工具,查看bit的值,可以看出其二進制值與我們操作的一致

image.png

位圖應(yīng)用之簽到

在很多時候,我們遇到用戶簽到的場景,用戶進入應(yīng)用時,獲取用戶當天的簽到情況,如果沒有簽到,用戶可以簽到,一般這種功能,可以通過set數(shù)據(jù)結(jié)構(gòu)或bitMap來實現(xiàn),但bitMap和set相比,其占用的空間更小,因此我們選擇使用bitMap來實現(xiàn)簽到的功能。
SignService:

package org.example.util;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.connection.BitFieldSubCommands;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;

import java.time.LocalDate;
import java.util.List;

@Service
public class SignService {
    @Autowired
    private RedisUtils redisUtils;

    @Autowired
    private RedisTemplate<String, Object> redisTemplate;

    /**
     * 簽到
     * @param id
     */
    public void sign(Integer id) {
        LocalDate now = LocalDate.now();
        String key = buildCacheKey(id, now);
        int dayOfMonth = now.getDayOfMonth();

        // 簽到
        redisUtils.setBit(key, dayOfMonth, true);
    }

    /**
     * 判斷是否簽到
     */
    public boolean isSign(Integer id) {
        LocalDate now = LocalDate.now();
        String key = buildCacheKey(id, now);
        int dayOfMonth = now.getDayOfMonth();
        return redisUtils.getBit(key, dayOfMonth);
    }

    /**
     * 獲取當月的簽到次數(shù)
     * @param id
     * @return
     */
    public Long getSignCountOfThisMonth(Integer id) {
        LocalDate now = LocalDate.now();
        String key = buildCacheKey(id, now);
        int dayOfMonth = now.getDayOfMonth();
        List<Long> result = redisTemplate.opsForValue().bitField(key,
                BitFieldSubCommands.create()
                        .get(BitFieldSubCommands.BitFieldType.unsigned(dayOfMonth)).valueAt(1));
        if (result == null || result.isEmpty()) {
            return 0L;
        }
        Long num = result.get(0);
        if (num == null || num == 0) {
            return 0L;
        }

        String binaryStr = Long.toString(num, 2);

        long count = 0;
        for (int i = 0; i < binaryStr.length(); i++) {
            char ch = binaryStr.charAt(i);
            if (ch == '1') {
                count ++;
            }
        }
        return count;
    }

    /**
     * 獲取本月連續(xù)簽到次數(shù)
     * @param id
     * @return
     */
    public Long getContinuousSignCountOfThisMonth(Integer id) {
        LocalDate now = LocalDate.now();
        String key = buildCacheKey(id, now);
        int dayOfMonth = now.getDayOfMonth();
        List<Long> result = redisTemplate.opsForValue().bitField(key,
                BitFieldSubCommands.create()
                        .get(BitFieldSubCommands.BitFieldType.unsigned(dayOfMonth)).valueAt(1));
        if (result == null || result.isEmpty()) {
            return 0L;
        }
        Long num = result.get(0);
        if (num == null || num == 0) {
            return 0L;
        }

        long count = 0;
        while (true) {
            if ((num & 1) == 0) {
                break;
            } else {
                count ++;
            }
            num >>>= 1;
        }
        return count;
    }

    private String buildCacheKey(Integer id, LocalDate localDate) {
        int year = localDate.getYear();
        int monthValue = localDate.getMonthValue();
        String key = "sign:" + year + ":" + monthValue + ":" + id;
        return key;
    }
}

測試代碼如下:

@Autowired
    private SignService signService;

    @Test
    public void testSign() {
        // 簽到
        signService.sign(1);

        // 判斷是否簽到
        System.out.println("是否簽到:" + signService.isSign(1));

        // 獲取當月的簽到次數(shù)
        System.out.println("當月的簽到次數(shù):" + signService.getSignCountOfThisMonth(1));

        // 獲取當月的連續(xù)簽到次數(shù)
        System.out.println("當月連續(xù)簽到次數(shù):" + signService.getContinuousSignCountOfThisMonth(1));
    }

運行結(jié)果如下:

image.png

到此這篇關(guān)于Redis應(yīng)用之簽到的使用的文章就介紹到這了,更多相關(guān)Redis 簽到內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Redis中Bitmap的使用示例

    Redis中Bitmap的使用示例

    本文主要介紹了Redis中Bitmap的使用示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-05-05
  • redis中跳表zset的具體使用

    redis中跳表zset的具體使用

    Redis跳表zset是一種結(jié)合了跳表和有序集合的高效數(shù)據(jù)結(jié)構(gòu),適用于實現(xiàn)排序和大規(guī)模數(shù)據(jù)的快速查詢,本文主要介紹了redis中跳表zset的具體使用,感興趣的可以了解一下
    2024-01-01
  • 使用redis實現(xiàn)附近的人功能

    使用redis實現(xiàn)附近的人功能

    這篇文章主要介紹了使用redis實現(xiàn)附近的人,實現(xiàn)諸如附近的人這類依賴于地理位置信息的功能,本文通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下
    2021-09-09
  • Redis操作相關(guān)命令之查看、停止、啟動命令

    Redis操作相關(guān)命令之查看、停止、啟動命令

    這篇文章主要介紹了Redis操作相關(guān)命令之查看、停止、啟動命令,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • Redis源碼環(huán)境構(gòu)建過程詳解

    Redis源碼環(huán)境構(gòu)建過程詳解

    這篇文章主要介紹了Redis源碼環(huán)境構(gòu)建過程,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-07-07
  • Redis簡易延時隊列的實現(xiàn)示例

    Redis簡易延時隊列的實現(xiàn)示例

    在實際的業(yè)務(wù)場景中,經(jīng)常會遇到需要延時處理的業(yè)務(wù),本文就來介紹有下Redis簡易延時隊列的實現(xiàn)示例,具有一定的參考價值,感興趣的可以了解一下
    2023-12-12
  • 詳解redis分布式鎖(優(yōu)化redis分布式鎖的過程及Redisson使用)

    詳解redis分布式鎖(優(yōu)化redis分布式鎖的過程及Redisson使用)

    在分布式的開發(fā)中,以電商庫存的更新功能進行講解,在實際的應(yīng)用中相同功能的消費者是有多個的,這篇文章主要介紹了redis分布式鎖詳解(優(yōu)化redis分布式鎖的過程及Redisson使用),需要的朋友可以參考下
    2021-11-11
  • Jedis操作Redis實現(xiàn)模擬驗證碼發(fā)送功能

    Jedis操作Redis實現(xiàn)模擬驗證碼發(fā)送功能

    Redis是一個著名的key-value存儲系統(tǒng),也是nosql中的最常見的一種,這篇文章主要給大家介紹Jedis操作Redis實現(xiàn)模擬驗證碼發(fā)送功能,感興趣的朋友一起看看吧
    2021-09-09
  • Redis使用元素刪除的布隆過濾器來解決緩存穿透問題

    Redis使用元素刪除的布隆過濾器來解決緩存穿透問題

    本文主要介紹了Redis使用元素刪除的布隆過濾器來解決緩存穿透問題,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • 使用Docker部署Redis并配置持久化與密碼保護的詳細步驟

    使用Docker部署Redis并配置持久化與密碼保護的詳細步驟

    本文將詳細介紹如何使用 Docker 部署 Redis,并通過 redis.conf 配置文件實現(xiàn)數(shù)據(jù)持久化和密碼保護,適合在生產(chǎn)環(huán)境中使用,文章通過代碼示例講解的非常詳細,需要的朋友可以參考下
    2025-03-03

最新評論