婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Redis基于Bitmap實現(xiàn)用戶簽到功能

Redis基于Bitmap實現(xiàn)用戶簽到功能

熱門標(biāo)簽:超呼電話機器人 魔獸2青云地圖標(biāo)注 十堰營銷電銷機器人哪家便宜 日本中國地圖標(biāo)注 宿遷便宜外呼系統(tǒng)平臺 鄭州人工智能電銷機器人系統(tǒng) 山東外呼銷售系統(tǒng)招商 北京400電話辦理收費標(biāo)準(zhǔn) 貴州電銷卡外呼系統(tǒng)

很多應(yīng)用上都有用戶簽到的功能,尤其是配合積分系統(tǒng)一起使用。現(xiàn)在有以下需求:

  • 簽到1天得1積分,連續(xù)簽到2天得2積分,3天得3積分,3天以上均得3積分等。
  • 如果連續(xù)簽到中斷,則重置計數(shù),每月重置計數(shù)。
  • 顯示用戶某月的簽到次數(shù)和首次簽到時間。
  • 在日歷控件上展示用戶每月簽到,可以切換年月顯示。
  • ...

功能分析

對于用戶簽到數(shù)據(jù),如果直接采用數(shù)據(jù)庫存儲,當(dāng)出現(xiàn)高并發(fā)訪問時,對數(shù)據(jù)庫壓力會很大,例如雙十一簽到活動。這時候應(yīng)該采用緩存,以減輕數(shù)據(jù)庫的壓力,Redis是高性能的內(nèi)存數(shù)據(jù)庫,適用于這樣的場景。

如果采用String類型保存,當(dāng)用戶數(shù)量大時,內(nèi)存開銷就非常大。

如果采用集合類型保存,例如Set、Hash,查詢用戶某個范圍的數(shù)據(jù)時,查詢效率又不高。

Redis提供的數(shù)據(jù)類型BitMap(位圖),每個bit位對應(yīng)0和1兩個狀態(tài)。雖然內(nèi)部還是采用String類型存儲,但Redis提供了一些指令用于直接操作BitMap,可以把它看作一個bit數(shù)組,數(shù)組的下標(biāo)就是偏移量。

它的優(yōu)點是內(nèi)存開銷小,效率高且操作簡單,很適合用于簽到這類場景。缺點在于位計算和位表示數(shù)值的局限。如果要用位來做業(yè)務(wù)數(shù)據(jù)記錄,就不要在意value的值。

Redis提供了以下幾個指令用于操作BitMap:

命令 說明 可用版本 時間復(fù)雜度
SETBIT 對 key 所儲存的字符串值,設(shè)置或清除指定偏移量上的位(bit)。 >= 2.2.0 O(1)
GETBIT 對 key 所儲存的字符串值,獲取指定偏移量上的位(bit)。 >= 2.2.0 O(1)
BITCOUNT 計算給定字符串中,被設(shè)置為 1 的比特位的數(shù)量。 >= 2.6.0 O(N)
BITPOS 返回位圖中第一個值為 bit 的二進制位的位置。 >= 2.8.7 O(N)
BITOP 對一個或多個保存二進制位的字符串 key 進行位元操作。 >= 2.6.0 O(N)
BITFIELD BITFIELD 命令可以在一次調(diào)用中同時對多個位范圍進行操作。 >= 3.2.0 O(1)

考慮到每月要重置連續(xù)簽到次數(shù),最簡單的方式是按用戶每月存一條簽到數(shù)據(jù)。Key的格式為 u:sign:{uid}:{yyyMM},而Value則采用長度為4個字節(jié)的(32位)的BitMap(最大月份只有31天)。BitMap的每一位代表一天的簽到,1表示已簽,0表示未簽。

例如 u:sign:1225:202101 表示ID=1225的用戶在2021年1月的簽到記錄

# 用戶1月6號簽到
SETBIT u:sign:1225:202101 5 1 # 偏移量是從0開始,所以要把6減1

# 檢查1月6號是否簽到
GETBIT u:sign:1225:202101 5 # 偏移量是從0開始,所以要把6減1

# 統(tǒng)計1月份的簽到次數(shù)
BITCOUNT u:sign:1225:202101

# 獲取1月份前31天的簽到數(shù)據(jù)
BITFIELD u:sign:1225:202101 get u31 0

# 獲取1月份首次簽到的日期
BITPOS u:sign:1225:202101 1 # 返回的首次簽到的偏移量,加上1即為當(dāng)月的某一天

示例代碼

using StackExchange.Redis;
using System;
using System.Collections.Generic;
using System.Linq;

/**
* 基于Redis Bitmap的用戶簽到功能實現(xiàn)類
* 
* 實現(xiàn)功能:
* 1. 用戶簽到
* 2. 檢查用戶是否簽到
* 3. 獲取當(dāng)月簽到次數(shù)
* 4. 獲取當(dāng)月連續(xù)簽到次數(shù)
* 5. 獲取當(dāng)月首次簽到日期
* 6. 獲取當(dāng)月簽到情況
*/
public class UserSignDemo
{
    private IDatabase _db;

    public UserSignDemo(IDatabase db)
    {
        _db = db;
    }

    /**
     * 用戶簽到
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return 之前的簽到狀態(tài)
     */
    public bool DoSign(int uid, DateTime date)
    {
        int offset = date.Day - 1;
        return _db.StringSetBit(BuildSignKey(uid, date), offset, true);
    }

    /**
     * 檢查用戶是否簽到
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return 當(dāng)前的簽到狀態(tài)
     */
    public bool CheckSign(int uid, DateTime date)
    {
        int offset = date.Day - 1;
        return _db.StringGetBit(BuildSignKey(uid, date), offset);
    }

    /**
     * 獲取用戶簽到次數(shù)
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return 當(dāng)前的簽到次數(shù)
     */
    public long GetSignCount(int uid, DateTime date)
    {
        return _db.StringBitCount(BuildSignKey(uid, date));
    }

    /**
     * 獲取當(dāng)月連續(xù)簽到次數(shù)
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return 當(dāng)月連續(xù)簽到次數(shù)
     */
    public long GetContinuousSignCount(int uid, DateTime date)
    {
        int signCount = 0;
        string type = $"u{date.Day}";   // 取1號到當(dāng)天的簽到狀態(tài)

        RedisResult result = _db.Execute("BITFIELD", (RedisKey)BuildSignKey(uid, date), "GET", type, 0);
        if (!result.IsNull)
        {
            var list = (long[])result;
            if (list.Length > 0)
            {
                // 取低位連續(xù)不為0的個數(shù)即為連續(xù)簽到次數(shù),需考慮當(dāng)天尚未簽到的情況
                long v = list[0];
                for (int i = 0; i  date.Day; i++)
                {
                    if (v >> 1  1 == v)
                    {
                        // 低位為0且非當(dāng)天說明連續(xù)簽到中斷了
                        if (i > 0) break;
                    }
                    else
                    {
                        signCount += 1;
                    }
                    v >>= 1;
                }
            }
        }
        return signCount;
    }

    /**
     * 獲取當(dāng)月首次簽到日期
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return 首次簽到日期
     */
    public DateTime? GetFirstSignDate(int uid, DateTime date)
    {
        long pos = _db.StringBitPosition(BuildSignKey(uid, date), true);
        return pos  0 ? null : date.AddDays(date.Day - (int)(pos + 1));
    }

    /**
     * 獲取當(dāng)月簽到情況
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return Key為簽到日期,Value為簽到狀態(tài)的Map
     */
    public Dictionarystring, bool> GetSignInfo(int uid, DateTime date)
    {
        Dictionarystring, bool> signMap = new Dictionarystring, bool>(date.Day);
        string type = $"u{GetDayOfMonth(date)}";
        RedisResult result = _db.Execute("BITFIELD", (RedisKey)BuildSignKey(uid, date), "GET", type, 0);
        if (!result.IsNull)
        {
            var list = (long[])result;
            if (list.Length > 0)
            {
                // 由低位到高位,為0表示未簽,為1表示已簽
                long v = list[0];
                for (int i = GetDayOfMonth(date); i > 0; i--)
                {
                    DateTime d = date.AddDays(i - date.Day);
                    signMap.Add(FormatDate(d, "yyyy-MM-dd"), v >> 1  1 != v);
                    v >>= 1;
                }
            }
        }
        return signMap;
    }

    private static string FormatDate(DateTime date)
    {
        return FormatDate(date, "yyyyMM");
    }

    private static string FormatDate(DateTime date, string pattern)
    {
        return date.ToString(pattern);
    }

    /**
     * 構(gòu)建簽到Key
     *
     * @param uid  用戶ID
     * @param date 日期
     * @return 簽到Key
     */
    private static string BuildSignKey(int uid, DateTime date)
    {
        return $"u:sign:{uid}:{FormatDate(date)}";
    }

    /**
     * 獲取月份天數(shù)
     *
     * @param date 日期
     * @return 天數(shù)
     */
    private static int GetDayOfMonth(DateTime date)
    {
        if (date.Month == 2)
        {
            return 28;
        }
        if (new int[] { 1, 3, 5, 7, 8, 10, 12 }.Contains(date.Month))
        {
            return 31;
        }
        return 30;
    }

    static void Main(string[] args)
    {
        ConnectionMultiplexer connection = ConnectionMultiplexer.Connect("192.168.0.104:7001,password=123456");

        UserSignDemo demo = new UserSignDemo(connection.GetDatabase());
        DateTime today = DateTime.Now;
        int uid = 1225;

        { // doSign
            bool signed = demo.DoSign(uid, today);
            if (signed)
            {
                Console.WriteLine("您已簽到:" + FormatDate(today, "yyyy-MM-dd"));
            }
            else
            {
                Console.WriteLine("簽到完成:" + FormatDate(today, "yyyy-MM-dd"));
            }
        }

        { // checkSign
            bool signed = demo.CheckSign(uid, today);
            if (signed)
            {
                Console.WriteLine("您已簽到:" + FormatDate(today, "yyyy-MM-dd"));
            }
            else
            {
                Console.WriteLine("尚未簽到:" + FormatDate(today, "yyyy-MM-dd"));
            }
        }

        { // getSignCount
            long count = demo.GetSignCount(uid, today);
            Console.WriteLine("本月簽到次數(shù):" + count);
        }

        { // getContinuousSignCount
            long count = demo.GetContinuousSignCount(uid, today);
            Console.WriteLine("連續(xù)簽到次數(shù):" + count);
        }

        { // getFirstSignDate
            DateTime? date = demo.GetFirstSignDate(uid, today);
            if (date.HasValue)
            {
                Console.WriteLine("本月首次簽到:" + FormatDate(date.Value, "yyyy-MM-dd"));
            }
            else
            {
                Console.WriteLine("本月首次簽到:無");
            }
        }

        { // getSignInfo
            Console.WriteLine("當(dāng)月簽到情況:");
            Dictionarystring, bool> signInfo = new Dictionarystring, bool>(demo.GetSignInfo(uid, today));
            foreach (var entry in signInfo)
            {
                Console.WriteLine(entry.Key + ": " + (entry.Value ? "√" : "-"));
            }
        }
    }
}

運行結(jié)果

 

更多應(yīng)用場景

  • 統(tǒng)計活躍用戶:把日期作為Key,把用戶ID作為offset,1表示當(dāng)日活躍,0表示當(dāng)日不活躍。還能使用位計算得到日活、月活、留存率等數(shù)據(jù)。
  • 用戶在線狀態(tài):跟統(tǒng)計活躍用戶一樣。

總結(jié)

  • 位圖優(yōu)點是內(nèi)存開銷小,效率高且操作簡單;缺點是位計算和位表示數(shù)值的局限。
  • 位圖適合二元狀態(tài)的場景,例如用戶簽到、在線狀態(tài)等場景。
  • String類型最大長度為512M。 注意SETBIT時的偏移量,當(dāng)偏移量很大時,可能會有較大耗時。 位圖不是絕對的好,有時可能更浪費空間。
  • 如果位圖很大,建議分拆鍵。如果要使用BITOP,建議讀取到客戶端再進行位計算。

參考資料

基于Redis位圖實現(xiàn)用戶簽到功能

Redis 深度歷險:核心原理與應(yīng)用實踐

Redis:Bitmap的setbit,getbit,bitcount,bitop等使用與應(yīng)用場景

BITFIELD SET command is not working

到此這篇關(guān)于Redis基于Bitmap實現(xiàn)用戶簽到功能的文章就介紹到這了,更多相關(guān)Redis Bitmap用戶簽到內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 基于Redis位圖實現(xiàn)用戶簽到功能
  • java redis 實現(xiàn)簡單的用戶簽到功能

標(biāo)簽:大慶 江蘇 臺州 朝陽 果洛 吉安 楊凌 北京

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Redis基于Bitmap實現(xiàn)用戶簽到功能》,本文關(guān)鍵詞  Redis,基于,Bitmap,實現(xiàn),用戶,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Redis基于Bitmap實現(xiàn)用戶簽到功能》相關(guān)的同類信息!
  • 本頁收集關(guān)于Redis基于Bitmap實現(xiàn)用戶簽到功能的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av
    色婷婷综合五月| 91精品国产综合久久婷婷香蕉 | 麻豆91精品视频| 在线不卡中文字幕| 五月婷婷综合网| 欧美精品日韩综合在线| 蜜臀av一区二区在线观看| 日韩欧美国产小视频| 国产资源在线一区| 国产免费久久精品| 色成人在线视频| 男女男精品网站| 欧美国产欧美亚州国产日韩mv天天看完整| 国产大陆精品国产| 亚洲欧美日韩久久| 正在播放亚洲一区| 国内成人免费视频| 日韩美女视频一区二区 | 精品国产网站在线观看| 成人精品电影在线观看| 亚洲精品中文字幕乱码三区| 91精品国产91综合久久蜜臀| 国产美女在线精品| 亚洲五月六月丁香激情| 精品欧美一区二区久久| 91精品1区2区| 国产精品一区二区在线看| 夜夜嗨av一区二区三区网页| www亚洲一区| 欧美专区亚洲专区| 国产aⅴ精品一区二区三区色成熟| 中文字幕一区二区三区四区不卡 | 欧美丰满高潮xxxx喷水动漫| 国内精品国产三级国产a久久| 亚洲欧美色综合| 精品伦理精品一区| 欧美日韩亚洲综合在线| 成人国产精品免费网站| 免费观看一级特黄欧美大片| 一区二区三区小说| 中文子幕无线码一区tr| 日韩三级高清在线| 精品污污网站免费看| av在线一区二区三区| 精一区二区三区| 青青草国产精品亚洲专区无| 亚洲激情在线激情| 综合欧美亚洲日本| 中文字幕一区二区三区在线播放| 日韩精品一区二区三区在线| 欧美日韩免费电影| 欧美亚洲综合久久| 在线影院国内精品| 91香蕉视频黄| 9i看片成人免费高清| 国产麻豆成人精品| 国产九九视频一区二区三区| 紧缚捆绑精品一区二区| 国产99久久久精品| 麻豆成人在线观看| 日韩黄色小视频| 亚洲一区视频在线观看视频| 亚洲你懂的在线视频| 伊人夜夜躁av伊人久久| 一区二区三区四区精品在线视频| 中文字幕第一区综合| 国产丝袜美腿一区二区三区| 久久亚洲精品小早川怜子| 日韩精品专区在线影院观看| 欧美成人一区二区三区在线观看 | 精品视频1区2区3区| 日本高清不卡aⅴ免费网站| 99riav一区二区三区| 99久久99久久精品免费观看| 色综合欧美在线视频区| 欧美日韩二区三区| 91精品国产福利| 久久在线免费观看| 国产精品嫩草影院com| 亚洲人成伊人成综合网小说| 亚洲欧美日韩中文字幕一区二区三区| 亚洲一二三四区| 麻豆国产精品视频| 成人理论电影网| 日本韩国视频一区二区| 欧美人xxxx| 国产色一区二区| 亚洲一区二区偷拍精品| 免费成人你懂的| 成人精品鲁一区一区二区| 一本一道波多野结衣一区二区| 欧洲生活片亚洲生活在线观看| 日韩你懂的在线观看| 国产精品午夜在线观看| 午夜精品免费在线| 韩国三级在线一区| 在线看国产一区二区| 久久久亚洲精华液精华液精华液| 国产精品传媒在线| 久久精品久久综合| 一本色道综合亚洲| 久久久久久久久一| 婷婷一区二区三区| 99视频有精品| 欧美不卡一区二区| 亚洲1区2区3区视频| 国产精品99久久久久久有的能看| 欧美午夜片在线看| 国产精品视频在线看| 看电影不卡的网站| 在线观看日韩av先锋影音电影院| 精品国偷自产国产一区| 亚洲一区二区三区四区在线观看 | 久久国产三级精品| 欧美性猛片aaaaaaa做受| 国产欧美日韩视频在线观看| 另类小说视频一区二区| 欧美日韩激情一区| 亚洲在线中文字幕| 一本一道综合狠狠老| 国产精品福利一区二区| 美女爽到高潮91| 欧美高清视频一二三区| 自拍偷在线精品自拍偷无码专区| 国产一区激情在线| 日韩精品一区二区三区四区视频| 亚洲福中文字幕伊人影院| 蜜臀av性久久久久蜜臀aⅴ四虎| 欧洲精品视频在线观看| 亚洲人成网站精品片在线观看| 国产成人精品一区二区三区网站观看| 欧美大度的电影原声| 免费在线一区观看| 日韩免费看网站| 久久精品99国产精品| 日韩欧美一区中文| 免费欧美高清视频| 26uuu久久天堂性欧美| 国产精品自拍毛片| 亚洲国产精品99久久久久久久久| 国产高清精品在线| 中文字幕巨乱亚洲| 粉嫩绯色av一区二区在线观看| 中文字幕久久午夜不卡| www.久久精品| 亚洲女人****多毛耸耸8| 91福利国产成人精品照片| 亚洲视频一区在线| 在线观看不卡一区| 丝袜诱惑制服诱惑色一区在线观看| 欧美日韩精品久久久| 亚洲欧美综合另类在线卡通| 国产91精品久久久久久久网曝门 | 日韩中文字幕不卡| 欧美高清视频一二三区| 极品美女销魂一区二区三区| 国产亚洲精品福利| 成人激情开心网| 一区二区三区在线观看视频| 7777精品久久久大香线蕉| 精品一区二区日韩| 国产日韩精品一区二区三区 | 91亚洲国产成人精品一区二区三 | 亚洲一区二区欧美| 色呦呦网站一区| 亚洲免费观看高清完整版在线| 精品污污网站免费看| 另类中文字幕网| 亚洲少妇屁股交4| 6080yy午夜一二三区久久| 老司机精品视频导航| 日韩精品一区二| zzijzzij亚洲日本少妇熟睡| 亚洲bdsm女犯bdsm网站| 久久久久国产精品麻豆ai换脸 | 国产激情视频一区二区三区欧美 | 成人午夜精品在线| 一区二区三区日韩欧美| 精品国精品国产| 在线日韩一区二区| 国产一区二区三区不卡在线观看| 欧美激情综合网| 欧美日免费三级在线| 成人美女视频在线观看18| 日韩福利电影在线| 日韩一区中文字幕| 日韩欧美精品三级| 欧美日韩在线综合| 色综合中文字幕| 国产成人aaaa| 久久精品国产亚洲高清剧情介绍| 亚洲欧美日韩在线| 久久精品视频网| 日韩写真欧美这视频| 欧美性生活大片视频| 99久久免费国产| 国产suv精品一区二区6| 激情五月播播久久久精品| 日产欧产美韩系列久久99| 中文久久乱码一区二区|