基于Redis位圖實(shí)現(xiàn)用戶簽到功能
場(chǎng)景需求
適用場(chǎng)景如簽到送積分、簽到領(lǐng)取獎(jiǎng)勵(lì)等,大致需求如下:
- 簽到1天送1積分,連續(xù)簽到2天送2積分,3天送3積分,3天以上均送3積分等。
- 如果連續(xù)簽到中斷,則重置計(jì)數(shù),每月初重置計(jì)數(shù)。
- 當(dāng)月簽到滿3天領(lǐng)取獎(jiǎng)勵(lì)1,滿5天領(lǐng)取獎(jiǎng)勵(lì)2,滿7天領(lǐng)取獎(jiǎng)勵(lì)3……等等。
- 顯示用戶某個(gè)月的簽到次數(shù)和首次簽到時(shí)間。
- 在日歷控件上展示用戶每月簽到情況,可以切換年月顯示……等等。
設(shè)計(jì)思路
對(duì)于用戶簽到數(shù)據(jù),如果每條數(shù)據(jù)都用K/V的方式存儲(chǔ),當(dāng)用戶量大的時(shí)候內(nèi)存開(kāi)銷是非常大的。而位圖(BitMap)是由一組bit位組成的,每個(gè)bit位對(duì)應(yīng)0和1兩個(gè)狀態(tài),雖然內(nèi)部還是采用String類型存儲(chǔ),但Redis提供了一些指令用于直接操作位圖,可以把它看作是一個(gè)bit數(shù)組,數(shù)組的下標(biāo)就是偏移量。它的優(yōu)點(diǎn)是內(nèi)存開(kāi)銷小、效率高且操作簡(jiǎn)單,很適合用于簽到這類場(chǎng)景。
Redis提供了以下幾個(gè)指令用于操作位圖:
考慮到每月初需要重置連續(xù)簽到次數(shù),最簡(jiǎn)單的方式是按用戶每月存一條簽到數(shù)據(jù)(也可以每年存一條數(shù)據(jù))。Key的格式為u:sign:uid:yyyyMM
,Value則采用長(zhǎng)度為4個(gè)字節(jié)(32位)的位圖(最大月份只有31天)。位圖的每一位代表一天的簽到,1表示已簽,0表示未簽。
例如u:sign:1000:201902
表示ID=1000的用戶在2019年2月的簽到記錄。
# 用戶2月17號(hào)簽到 SETBIT u:sign:1000:201902 16 1 # 偏移量是從0開(kāi)始,所以要把17減1 # 檢查2月17號(hào)是否簽到 GETBIT u:sign:1000:201902 16 # 偏移量是從0開(kāi)始,所以要把17減1 # 統(tǒng)計(jì)2月份的簽到次數(shù) BITCOUNT u:sign:1000:201902 # 獲取2月份前28天的簽到數(shù)據(jù) BITFIELD u:sign:1000:201902 get u28 0 # 獲取2月份首次簽到的日期 BITPOS u:sign:1000:201902 1 # 返回的首次簽到的偏移量,加上1即為當(dāng)月的某一天
示例代碼
import redis.clients.jedis.Jedis; import java.time.LocalDate; import java.time.format.DateTimeFormatter; import java.util.HashMap; import java.util.List; import java.util.Map; import java.util.TreeMap; /** * 基于Redis位圖的用戶簽到功能實(shí)現(xiàn)類 * <p> * 實(shí)現(xiàn)功能: * 1. 用戶簽到 * 2. 檢查用戶是否簽到 * 3. 獲取當(dāng)月簽到次數(shù) * 4. 獲取當(dāng)月連續(xù)簽到次數(shù) * 5. 獲取當(dāng)月首次簽到日期 * 6. 獲取當(dāng)月簽到情況 */ public class UserSignDemo { private Jedis jedis = new Jedis(); /** * 用戶簽到 * * @param uid 用戶ID * @param date 日期 * @return 之前的簽到狀態(tài) */ public boolean doSign(int uid, LocalDate date) { int offset = date.getDayOfMonth() - 1; return jedis.setbit(buildSignKey(uid, date), offset, true); } /** * 檢查用戶是否簽到 * * @param uid 用戶ID * @param date 日期 * @return 當(dāng)前的簽到狀態(tài) */ public boolean checkSign(int uid, LocalDate date) { int offset = date.getDayOfMonth() - 1; return jedis.getbit(buildSignKey(uid, date), offset); } /** * 獲取用戶簽到次數(shù) * * @param uid 用戶ID * @param date 日期 * @return 當(dāng)前的簽到次數(shù) */ public long getSignCount(int uid, LocalDate date) { return jedis.bitcount(buildSignKey(uid, date)); } /** * 獲取當(dāng)月連續(xù)簽到次數(shù) * * @param uid 用戶ID * @param date 日期 * @return 當(dāng)月連續(xù)簽到次數(shù) */ public long getContinuousSignCount(int uid, LocalDate date) { int signCount = 0; String type = String.format("u%d", date.getDayOfMonth()); List<Long> list = jedis.bitfield(buildSignKey(uid, date), "GET", type, "0"); if (list != null && list.size() > 0) { // 取低位連續(xù)不為0的個(gè)數(shù)即為連續(xù)簽到次數(shù),需考慮當(dāng)天尚未簽到的情況 long v = list.get(0) == null ? 0 : list.get(0); for (int i = 0; i < date.getDayOfMonth(); i++) { if (v >> 1 << 1 == v) { // 低位為0且非當(dāng)天說(shuō)明連續(xù)簽到中斷了 if (i > 0) break; } else { signCount += 1; } v >>= 1; } } return signCount; } /** * 獲取當(dāng)月首次簽到日期 * * @param uid 用戶ID * @param date 日期 * @return 首次簽到日期 */ public LocalDate getFirstSignDate(int uid, LocalDate date) { long pos = jedis.bitpos(buildSignKey(uid, date), true); return pos < 0 ? null : date.withDayOfMonth((int) (pos + 1)); } /** * 獲取當(dāng)月簽到情況 * * @param uid 用戶ID * @param date 日期 * @return Key為簽到日期,Value為簽到狀態(tài)的Map */ public Map<String, Boolean> getSignInfo(int uid, LocalDate date) { Map<String, Boolean> signMap = new HashMap<>(date.getDayOfMonth()); String type = String.format("u%d", date.lengthOfMonth()); List<Long> list = jedis.bitfield(buildSignKey(uid, date), "GET", type, "0"); if (list != null && list.size() > 0) { // 由低位到高位,為0表示未簽,為1表示已簽 long v = list.get(0) == null ? 0 : list.get(0); for (int i = date.lengthOfMonth(); i > 0; i--) { LocalDate d = date.withDayOfMonth(i); signMap.put(formatDate(d, "yyyy-MM-dd"), v >> 1 << 1 != v); v >>= 1; } } return signMap; } private static String formatDate(LocalDate date) { return formatDate(date, "yyyyMM"); } private static String formatDate(LocalDate date, String pattern) { return date.format(DateTimeFormatter.ofPattern(pattern)); } private static String buildSignKey(int uid, LocalDate date) { return String.format("u:sign:%d:%s", uid, formatDate(date)); } public static void main(String[] args) { UserSignDemo demo = new UserSignDemo(); LocalDate today = LocalDate.now(); { // doSign boolean signed = demo.doSign(1000, today); if (signed) { System.out.println("您已簽到:" + formatDate(today, "yyyy-MM-dd")); } else { System.out.println("簽到完成:" + formatDate(today, "yyyy-MM-dd")); } } { // checkSign boolean signed = demo.checkSign(1000, today); if (signed) { System.out.println("您已簽到:" + formatDate(today, "yyyy-MM-dd")); } else { System.out.println("尚未簽到:" + formatDate(today, "yyyy-MM-dd")); } } { // getSignCount long count = demo.getSignCount(1000, today); System.out.println("本月簽到次數(shù):" + count); } { // getContinuousSignCount long count = demo.getContinuousSignCount(1000, today); System.out.println("連續(xù)簽到次數(shù):" + count); } { // getFirstSignDate LocalDate date = demo.getFirstSignDate(1000, today); System.out.println("本月首次簽到:" + formatDate(date, "yyyy-MM-dd")); } { // getSignInfo System.out.println("當(dāng)月簽到情況:"); Map<String, Boolean> signInfo = new TreeMap<>(demo.getSignInfo(1000, today)); for (Map.Entry<String, Boolean> entry : signInfo.entrySet()) { System.out.println(entry.getKey() + ": " + (entry.getValue() ? "√" : "-")); } } } }
運(yùn)行結(jié)果
您已簽到:2019-02-18
您已簽到:2019-02-18
本月簽到次數(shù):11
連續(xù)簽到次數(shù):8
本月首次簽到:2019-02-02
當(dāng)月簽到情況:
2019-02-01: -
2019-02-02: √
2019-02-03: √
2019-02-04: -
2019-02-05: -
2019-02-06: √
2019-02-07: -
2019-02-08: -
2019-02-09: -
2019-02-10: -
2019-02-11: √
2019-02-12: √
2019-02-13: √
2019-02-14: √
2019-02-15: √
2019-02-16: √
2019-02-17: √
2019-02-18: √
2019-02-19: -
2019-02-20: -
2019-02-21: -
2019-02-22: -
2019-02-23: -
2019-02-24: -
2019-02-25: -
2019-02-26: -
2019-02-27: -
2019-02-28: -
參考鏈接
Redis 深度歷險(xiǎn):核心原理與應(yīng)用實(shí)踐
到此這篇關(guān)于基于Redis位圖實(shí)現(xiàn)用戶簽到功能的文章就介紹到這了,更多相關(guān)Redis用戶簽到內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- Redis應(yīng)用之簽到的使用
- SpringBoot+Redis?BitMap實(shí)現(xiàn)簽到與統(tǒng)計(jì)的項(xiàng)目實(shí)踐
- PHP利用redis位圖實(shí)現(xiàn)簡(jiǎn)單的簽到功能
- 微服務(wù)?Spring?Boot?整合?Redis?BitMap?實(shí)現(xiàn)?簽到與統(tǒng)計(jì)功能
- Redis基于Bitmap實(shí)現(xiàn)用戶簽到功能
- java redis 實(shí)現(xiàn)簡(jiǎn)單的用戶簽到功能
- PHP使用redis位圖bitMap 實(shí)現(xiàn)簽到功能
- Redis實(shí)現(xiàn)每日簽到功能(大數(shù)據(jù)量)
相關(guān)文章
Linux服務(wù)器快速安裝Redis6.0步驟示例詳解
這篇文章主要為大家介紹了Linux服務(wù)器快速安裝Redis6.0步驟示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12Redis創(chuàng)建并修改Lua 環(huán)境的實(shí)現(xiàn)方法
為了在Redis服務(wù)器中執(zhí)行Lua腳本, Redis在服務(wù)器內(nèi)嵌了一個(gè)Lua環(huán)境, 并對(duì)這個(gè)Lua環(huán)境進(jìn)行了一系列修改,本文主要介紹了Redis創(chuàng)建并修改Lua 環(huán)境的實(shí)現(xiàn)方法,具有一定的參考價(jià)值,感興趣的可以了解一下2024-05-05redis+mysql+quartz 一種紅包發(fā)送功能的實(shí)現(xiàn)
這篇文章主要介紹了redis+mysql+quartz 一種紅包發(fā)送功能的實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下2017-01-01Redisson分布式限流的實(shí)現(xiàn)原理解析
這篇文章主要為大家介紹了Redisson分布式限流的實(shí)現(xiàn)原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02