Redis:用户签到和UV统计

一、用户签到:BitMap功能

代表某个用户某一天的签到记录;如果有1000万用户,平均没人每年签到10次,一年则为1亿条;数据庞大;也耗内存;

Redis:用户签到和UV统计_第1张图片
Redis:用户签到和UV统计_第2张图片
Redis:用户签到和UV统计_第3张图片

二、用户签到:实现签到功能

Redis:用户签到和UV统计_第4张图片

/**
     * 用户签到功能
     * @return
     */
    @Override
    public Result sign() {
        // 获取用户当前
        UserDTO user = UserHolder.getUser();
        Long userId = user.getId();
        // 拿到日期
        LocalDateTime now = LocalDateTime.now();
        // 拼接key
        String keySuffix = now.format(DateTimeFormatter.ofPattern(":yyyyMM"));
        String key = com.hmdp.utils.RedisConstants.USER_SIGN_KEY + userId + keySuffix;
        // 拿到当前是这个月的第几天
        int dayOfMonth = now.getDayOfMonth();
        // 写入redis setbit key offset 1
        redisTemplate.opsForValue().setBit(key,dayOfMonth - 1,true);
        return Result.ok();
    }

三、连续签到统计

Redis:用户签到和UV统计_第5张图片

 /**
     * 统计连续签到天数
     * @return
     */
    @Override
    public Result signCount() {
        // 1.获取当前登录用户
        Long userId = UserHolder.getUser().getId();
        // 2.获取日期
        LocalDateTime now = LocalDateTime.now();
        // 3.拼接key
        String keySuffix = now.format(DateTimeFormatter.ofPattern(":yyyyMM"));
        String key = USER_SIGN_KEY + userId + keySuffix;
        // 4.获取今天是本月的第几天
        int dayOfMonth = now.getDayOfMonth();
        // 5.获取本月截止今天为止的所有的签到记录,返回的是一个十进制的数字 BITFIELD sign:5:202203 GET u14 0
        List<Long> result = redisTemplate.opsForValue().bitField(
                key,
                BitFieldSubCommands.create()
                        .get(BitFieldSubCommands.BitFieldType.unsigned(dayOfMonth)).valueAt(0)
        );
        if (result == null || result.isEmpty()) {
            // 没有任何签到结果
            return Result.ok(0);
        }
        Long num = result.get(0);
        if (num == null || num == 0) {
            return Result.ok(0);
        }
        // 6.循环遍历
        int count = 0;
        while (true) {
            // 6.1.让这个数字与1做与运算,得到数字的最后一个bit位  // 判断这个bit位是否为0
            if ((num & 1) == 0) {
                // 如果为0,说明未签到,结束
                break;
            }else {
                // 如果不为0,说明已签到,计数器+1
                count++;
            }
            // 把数字右移一位,抛弃最后一个bit位,继续下一个bit位
            num >>>= 1;
        }
        return Result.ok(count);
    }

四、UV统计

UV:unique vistitor :独立访客量,指通过互联网访问、浏览网页的自然人。1天内同一个用户多次访问该网站,只记录1次。
PV:page view.页面访问量或者点击量,用户每访问一个页面,记录1次叫pv。往往用来衡量网站的流量。
Redis:用户签到和UV统计_第6张图片

五、UV统计实现:百万数据

@Test
    void testHyperLogLog() {
        String[] values = new String[1000];
        int j = 0;
        for (int i = 0; i < 1000000; i++) {
            j = i % 1000;
            values[j] = "user_" + i;
            if(j == 999){
                // 发送到Redis
                stringRedisTemplate.opsForHyperLogLog().add("hl2", values);
            }
        }
        // 统计数量
        Long count = stringRedisTemplate.opsForHyperLogLog().size("hl2");
        System.out.println("count = " + count);
    }

你可能感兴趣的:(java)