【Redis】签到点赞和UV统计

Redis签到点赞和UV统计

点赞

点赞功能分析

需求:

  1. 同一个用户只能点赞一次,再次点击则取消点赞
  2. 如果当前用户已经点赞,则点赞按钮高亮显示(前端判断字段isLike属性)

实现步骤:

  1. 利用Redis的set集合判断是否点赞过,将用户id保存到set中
  2. 判断当前登录用户是否点赞过,赋值给isLike字段
  3. 通过Redis的set集合中Scard命令获取成员个数,即点赞次数

业务实现

LikedDTO

@Data
@AllArgsConstructor
@NoArgsConstructor
public class LikedDTO {/*** 点赞数量*/long likedSum;/*** 用户是否点过赞*/Boolean isLiked;
}

点赞操作

// 点赞操作
@Override
public String doLike() {String key = "RedisSessionDemo:liked";String phone = UserHolder.getUser().getPhone();// 查询是否点赞过Boolean isLiked = redisTemplate.opsForSet().isMember(key, phone);if (BooleanUtil.isTrue(isLiked)) {// 点赞过 -> 取消点赞redisTemplate.opsForSet().remove(key, phone);return "取消点赞成功";}// 没点赞过 -> 点赞redisTemplate.opsForSet().add(key, phone);return "点赞成功";
}

获取点赞数据

// 获取点赞数据
@Override
public LikedDTO getLiked() {String key = "RedisSessionDemo:liked";Long likedNum = redisTemplate.opsForSet().size(key);if (likedNum == null) {likedNum = 0L;}UserDTO user = UserHolder.getUser();Boolean isLiked = false;if (user != null) {isLiked = redisTemplate.opsForSet().isMember(key, user.getPhone());}return new LikedDTO(likedNum, isLiked);
}

点赞排行

功能分析

点赞排行:类似朋友圈的点赞列表,按照点赞的先后顺序展示头像等信息。

使用 sorted set 结构,将点赞的时间戳作为分数值记录。

功能实现

修改点赞函数

// 获取点赞数据
@Override
public LikedDTO getLiked2() {String key = "RedisSessionDemo:liked";Long likedNum = redisTemplate.opsForZSet().size(key);if (likedNum == null) {likedNum = 0L;}UserDTO user = UserHolder.getUser();boolean isLiked = false;if (user != null) {Double score = redisTemplate.opsForZSet().score(key, user.getPhone());isLiked = (score != null && score > 0);}return new LikedDTO(likedNum, isLiked);
}// 点赞操作
@Override
public String doLike2() {String key = "RedisSessionDemo:liked";String phone = UserHolder.getUser().getPhone();// 查询是否点赞过Double isLiked = redisTemplate.opsForZSet().score(key, phone);if (isLiked != null && isLiked > 0) {// 点赞过 -> 取消点赞redisTemplate.opsForZSet().remove(key, phone);return "取消点赞成功";}// 没点赞过 -> 点赞redisTemplate.opsForZSet().add(key, phone, System.currentTimeMillis());return "点赞成功";
}

获取点赞列表

// 获取点赞列表
@Override
public List<String> getLikedList() {String key = "RedisSessionDemo:liked";// 获取所有元素Set<String> set = redisTemplate.opsForZSet().range(key, 0, -1);if (set != null) {return new ArrayList<>(set);}return Collections.emptyList();
}

用户签到

BitMap用法

我们按月来统计用户签到信息,签到记录为1,未签到则记录为0。

把每一个bit位对应当月的每一天,形成了映射关系。用0和1标示业务状态,这种思路就称为位图(BitMap)。

Redis中是利用string类型数据结构实现BitMap,因此最大上限是512M,转换为bit则是 2 32 2^{32} 232 个bit位。

BitMap的操作命令有:

  1. SETBIT:向指定位置(offset)存入一个0或1
  2. GETBIT :获取指定位置(offset)的bit值
  3. BITCOUNT :统计BitMap中值为1的bit位的数量
  4. BITFIELD :操作(查询、修改、自增)BitMap中bit数组中的指定位置(offset)的值
  5. BITFIELD_RO :获取BitMap中bit数组,并以十进制形式返回
  6. BITOP :将多个BitMap的结果做位运算(与 、或、异或)
  7. BITPOS :查找bit数组中指定范围内第一个0或1出现的位置

实现签到功能

因为BitMap底层是基于String数据结构,因此其操作也都封装在字符串相关操作中了。

public Boolean sign() {String phone = UserHolder.getUser().getPhone();Date date = new Date();String yearAndMonth = new SimpleDateFormat("yyyy:MM").format(date);String key = "RedisSessionDemo:user:sign:" + phone + ":" + yearAndMonth;int day = Integer.parseInt(new SimpleDateFormat("DD").format(date));// 实现签到redisTemplate.opsForValue().setBit(key, day, true);return true;
}

签到统计

连续签到:从最后一次签到开始向前统计,直到遇到第一次未签到为止的签到次数

封装SignData类

@Data
@AllArgsConstructor
@NoArgsConstructor
public class SignData {// 月签到次数Integer MonthTimes;// 月连续签到次数Integer ContinuousTimes;
}

业务实现

@Override
public SignData signdata() {// 获取 bitmapString phone = UserHolder.getUser().getPhone();Date date = new Date();String yearAndMonth = new SimpleDateFormat("yyyy:MM").format(date);String key = "RedisSessionDemo:user:sign:" + phone + ":" + yearAndMonth;int day = Integer.parseInt(new SimpleDateFormat("DD").format(date));List<Long> list = redisTemplate.opsForValue().bitField(key, BitFieldSubCommands.create().get(BitFieldSubCommands.BitFieldType.unsigned(day + 1)).valueAt(0));if (list == null || list.isEmpty()) {return new SignData(0, 0);}Long sign = list.get(0);if (sign == null) {return new SignData(0, 0);}// 统计计算int MonthTimes = 0;int ContinuousTimes = 0;boolean isContinuous = true;while (sign != 0) {// 连续签到if (isContinuous) {if ((sign & 1) == 1) {ContinuousTimes++;} else {isContinuous = false;}}// 月签到次数if ((sign & 1) == 1) {MonthTimes++;}sign = sign >> 1;}return new SignData(MonthTimes, ContinuousTimes);
}

UV统计

HyperLogLog

  • UV:全称Unique Visitor,也叫独立访客量,是指通过互联网访问、浏览这个网页的自然人。1天内同一个用户多次访问该网站,只记录1次。
  • PV:全称Page View,也叫页面访问量或点击量,用户每访问网站的一个页面,记录1次PV,用户多次打开页面,则记录多次PV。往往用来衡量网站的流量。

UV统计在服务端做会比较麻烦,因为要判断该用户是否已经统计过了,需要将统计过的用户信息保存。但是如果每个访问的用户都保存到Redis中,数据量会非常恐怖。

Hyperloglog(HLL)是从Loglog算法派生的概率算法,用于确定非常大的集合的基数,而不需要存储其所有值。相关算法

Redis中的HLL是基于string结构实现的,单个HLL的内存永远小于16kb,内存占用低的令人发指!

作为代价,其测量结果是概率性的,有小于0.81%的误差。不过对于UV统计来说,这完全可以忽略。

  1. 作用:做海量数据的统计工作
  2. 优点:内存占用极低、性能非常好
  3. 缺点:有一定的误差

业务实现

@Test
void hyperlogTest() {for (int i = 0; i < 100; i++) {stringRedisTemplate.opsForHyperLogLog().add("hyperlogTest", "user-" + i);}Long size = stringRedisTemplate.opsForHyperLogLog().size("hyperlogTest");System.out.println(size);
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/661039.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于修改数据库服务器时间导致达梦数据库集群裂开

故障原因&#xff1a; 因为每天数据库服务器时间都不一致&#xff0c;想要给数据库服务器配置个NTP服务器。结果导致达梦数据库裂库。后面查看了达梦系统管理员手册了解了达梦集群的机制&#xff0c;知道数据库服务器时间需要先关闭数据库服务之后才可以修改数据库服务器时间。…

五款颠覆工作方式的工作软件

在当今快节奏的工作环境中&#xff0c;选择合适的工作软件是提高效率、实现协作的关键。感谢你的提问&#xff0c;下面我将为你介绍五款令人赞叹的工作软件&#xff0c;它们会让你的工作变得更有趣而富有成效&#xff01; 1. 亿可达 是我最近在用的一款软件连接器&#xff0c…

ubuntu16.04环境轻松安装和应用opencv4.9.0(基于源码编译)

目录 一、环境准备 1、安装cmake 2、安装依赖 3、从github上下载opencv4.9.0.zip 二、安装opencv4.9.0 1、解压4.9.0.zip 2、进入build目录编译 3、安装编译好的相关库 4、修改opencv配置文件并使其生效 5、添加PKG_CONFIG路径&#xff0c;并使其生效 三、opencv环境…

聚道云软件连接器:连接薪人薪事与携程商旅的桥梁,出差管理效率的新篇章

客户介绍 某科技有限公司是一家专注于提供数字化解决方案的高科技企业。公司拥有一支由业内资深专家和优秀工程师组成的团队&#xff0c;致力于为企业提供全方位的数字化服务。该公司拥有自主研发能力&#xff0c;拥有多项知识产权和专利技术&#xff0c;确保为客户提供技术领…

2024美赛数学建模A题思路源码

比赛当天第一时间更新&#xff01; 赛题目的 赛题目的&#xff1a; 问题描述&#xff1a; 解题的关键&#xff1a; 问题一. 问题分析 问题解答 问题二. 问题分析 问题解答 问题三. 问题分析 问题解答 问题四. 问题分析 问题解答 问题五. 问题分析 问题解答

数值函数

目录 四舍五入操作 测试四舍五入 截取小数(所有的小数都不进位) 求模(求余数) 求模操作 Oracle从入门到总裁:https://blog.csdn.net/weixin_67859959/article/details/135209645 数值函数可以对数字进行处理&#xff0c;常用的主要函数有 3 个&#xff1a; round()、trunc…

力扣刷题-169.多数元素

给定一个大小为 n 的数组 nums &#xff0c;返回其中的多数元素。多数元素是指在数组中出现次数 大于 ⌊ n/2 ⌋ 的元素。 你可以假设数组是非空的&#xff0c;并且给定的数组总是存在多数元素。 示例 1&#xff1a; 输入&#xff1a;nums [3,2,3] 输出&#xff1a;3 示例 …

Aigtek大功率信号源怎么使用的

大功率信号源是在实验室、测试和通信系统中经常使用的重要设备。它能够提供高功率的信号&#xff0c;用于驱动各种设备和系统。在使用大功率信号源时&#xff0c;有一些关键的步骤和指南&#xff0c;可以确保安全、有效地操作设备并获得稳定的输出。本文将详细介绍大功率信号源…

docker-compose部署开源培训系统playedu

docker-compose 安装 本文安装过程需要您准备以下环境&#xff1a; Dockerdocker-compose 第一步、下载 playedu-docker-compose 仓库 git clone -b 1.5.1 https://gitee.com/playeduxyz/compose.git playedu-docker-compose第二步、构建镜像 下面命令 # 开头的是对下一行命…

DevOps落地笔记-05|非功能需求:如何有效关注非功能需求

上一讲主要介绍了看板方法以及如何使用看板方法来解决软件研发过程中出现的团队过载、工作不均、任务延期等问题。通过学习前面几个课时介绍的知识&#xff0c;你的团队开始源源不断地交付用户价值。用户对交付的功能非常满意&#xff0c;但等到系统上线后经常出现服务不可用的…

C程序设计(第5版)谭浩强习题解答 第6章 利用数组处理批量数据

C程序设计(第5版)谭浩强习题解答 第6章 利用数组处理批量数据 1.用筛选法求100之内的素数 #include <stdio.h> #include <math.h> int main() {int i, j, n, a[101];for (i 1; i < 100; i)a[i] i;a[1] 0;for (i 2; i < sqrt(100); i)for (j i 1; j …

笔记--扩展欧几里得算法

AcWing.877.欧几里得算法 给定 n n n 对正整数 a a ai, b b bi&#xff0c;对于每对数&#xff0c;求出一组 x x xi, y y yi&#xff0c;使其满足 a a ai x x xi b b bi y y yi g c d ( a gcd(a gcd(ai , b ,b ,bi ) ) )。 输入格式 第一行包含整数 n n n。 接下来 …

【Spring框架】@Cacheable注解:缓存最佳实践

在Java开发中&#xff0c;性能优化是一个永恒的话题。对于使用Spring框架的应用程序来说&#xff0c;Cacheable 注解提供了一种简单有效的方式来提升性能&#xff0c;特别是对于那些计算成本高或数据变化不频繁的操作。本文将深入探讨 Cacheable 的使用方法和注意事项&#xff…

在Vue 3中,理解使用defineEmits函数来定义组件的事件。

在Vue 3中&#xff0c;可以使用defineEmits函数来定义组件的事件。defineEmits函数接受一个对象作为参数&#xff0c;该对象的键是事件名&#xff0c;值是一个回调函数或一个函数数组。这些回调函数将被组件中的$emit方法触发。 以下是一个示例&#xff1a; import { defineC…

C#使用OpenCvSharp4库中5个基础函数-灰度化、高斯模糊、Canny边缘检测、膨胀、腐蚀

C#使用OpenCvSharp4库中5个基础函数-灰度化、高斯模糊、Canny边缘检测、膨胀、腐蚀 使用OpenCV可以对彩色原始图像进行基本的处理&#xff0c;涉及到5个常用的处理&#xff1a; 灰度化 模糊处理 Canny边缘检测 膨胀 腐蚀 1、测试图像lena.jpg 本例中我们采用数字图像处…

Java玩转《啊哈算法》解密QQ号之队列

行有不得&#xff0c;反求诸己 文章目录 开头代码地址引子案例分析代码 队列封装升级演示 开头 各位好&#xff01;本人在看《啊哈算法》&#xff0c;写的确实不错。 但略微遗憾的是&#xff0c;书籍示例代码是c语言&#xff0c;不是本人常用的Java。 那就弥补遗憾&#xff…

【C语言】va_list(可变参数处理)

C 语言中的 va_list 类型允许函数接受可变数量的参数&#xff0c;这在编写需要处理不定数量参数的函数时非常有用。va_list 类型是在 stdarg.h 头文件中定义的&#xff0c;它允许函数处理可变数量的参数。下面我们将详细介绍 va_list 的用法以及实际应用示例。 一、va_list的用…

【力扣刷题练习】415. 字符串相加

题目描述&#xff1a; 给定两个字符串形式的非负整数 num1 和num2 &#xff0c;计算它们的和并同样以字符串形式返回。 你不能使用任何內建的用于处理大整数的库&#xff08;比如 BigInteger&#xff09;&#xff0c; 也不能直接将输入的字符串转换为整数形式。 题目解答&am…

分布式搜索引擎_学习笔记_3

分布式搜索引擎03 0.学习目标 1.数据聚合 **聚合&#xff08;aggregations&#xff09;**可以让我们极其方便的实现对数据的统计、分析、运算。例如&#xff1a; 什么品牌的手机最受欢迎&#xff1f;这些手机的平均价格、最高价格、最低价格&#xff1f;这些手机每月的销售…

Postgresql使用update

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 解决问题一、关联表更新1.关联一张表2.关联多张表 二、根据状态更新为不同的值 解决问题 通过多张关联表更新主表的字段&#xff0c;根据状态更新为不同的值。 一、…