【Redis】共同关注列表与基于Feed流的关注消息滚动分页推送的实现

news2024/12/29 10:15:38

目录

一、共同关注

1、思路

2、实现步骤

二、Feed流

1、概念

2、需求

3、TimeLine的三种模式

1.拉

2.推

3.推拉结合

4、TimeLine三种模式的区别

三、关注推送

1、需求

2、实现思路

3、Redis数据结构的选择

4、滚动分页

5、代码实现

1.博主

2.粉丝


一、共同关注

1、思路

当登录的用户点击共同关注时,需要展示当前页面的用户与登录用户直接的共同关注,此时我们可以在数据库中维护一张关注表里面记录了关注者与被关注者的id,我们可以直接拿着该用户id去查询登录他关注了哪些人,同样拿着当前访问主页用户的id去查询该用户的关注然后求交集即可,那么问题来了如何求交集呢?我们可以通过Redis中的set数据结构的求交集命令来处理

2、实现步骤

首先在项目中经常会有这样的需求就是实现关注于共同关注的功能,在我们实现关注的时候,首先我们将关注的信息存入数据库中,插入之后再将该关注信息以用户id作为redis中的key的组成,以被关注者的id作为value存入redis的set集合中,取消关注时删除数据库中关注信息后将被关注用户的id从set中移除,此时当用户访问到某个用户显示共同关注时会将访问的该用户的id传回服务器,此时服务器拿着被访问用户有的id与当前登录用户的id构造成两个set集合个key通过redis的 sinter 命令求交集获得共同关注的所有用户id,然后拿着id在从数据库中查询这些共同关注用户的信息返回给客户端

二、Feed流

1、概念

关注推送也叫Feed流,直译为投喂。为用户持续提供沉浸式体验,通过无限下拉刷新获取新的消息,他的实现主要有两种实现方式,一种是TimeLine方式,他不会做内容筛选,只是简单的按照内容发布的时间进行排序,常用于好友或关注,比如朋友圈,他的优点是信息全面不会缺少,实现起来相对简单,缺点就是消息噪音多,用户不一定感兴趣。还有一种是智能排序,利用智能算法屏蔽掉违规的、用户不感兴趣的信息,缺点是如果算法不准确,可能会起到反作用

2、需求

用户在平台发布新的内容,粉丝收到该消息

3、TimeLine的三种模式

1.拉

拉模式是粉丝相获取相应的内容时去每个关注者的发件箱里获取相应的内容。将所有的内容拉取下来

2.推

推模式是用户发布内容后将内容推送到每个粉丝的收件箱里,粉丝查看内容时只需要从收件箱里拿出来即可

3.推拉结合

推拉模式则是既推送也拉取,比如对于活跃的粉丝可以采用推模式,而对于普通的粉丝可以采用拉模式

4、TimeLine三种模式的区别

拉模式推模式推拉结合
写比例
读比例
用户读取延迟
实现难度复制简单很复杂
使用场景很少用户量少有千万级别用户量的大V

三、关注推送

1、需求

在项目中我们需要实现博主发布视频或者发布文章需要给用户将该信息进行推送,类似微信当朋友圈以及抖音,当用户发布视频后推送将该视频对其粉丝进行推送,此时我们可以基于上述Feed流来实现,当用户发布相关的消息后此时我们先将该消息存入数据库,后在将该消息id与当前时间戳作为value推送给每个粉丝的收件箱,当粉丝访问相应页面时直接从收件箱通过滚动排序拉取对应的消息,此时为什么不直接采用分页查询呢,如果使用传统的分页来实现时,比如此时有5条数据我们要根据按时间查最新的在最上面,此时也就是时间戳越大越新,此时我们查询第一页每页2条时,查询出来了5,4此时查询第二页之前收件箱里又来了一个消息马,此时消息下标发送变化导致4被重复查询出

2、实现思路

首先我需要在用户发布内容时将内容推送到每个粉丝的收件箱中,每个粉丝在访问相应页面时开始滚动查询此时刻的推送内容,当他刷新时滚动查询刷新时刻的内容,那么对于redis实现时的收件箱数据结构有什么要求呢:他要按照时间排序,所有有排序功能,且他实现分页得支持下标查询。

3、Redis数据结构的选择

reids中满足上述需要的有list与sortedSet,但是Feed流的数据是不断变化的,角标也是不断变化的,所以使用后者更合适

4、滚动分页

使用sortedSet实现的时候,使用zrevrangebyscore key max min limit offset count 命令来进行查询,根据时间戳的区间进行滚动分页查询,其中max是score的最大值,min是区间的最小值,offset是偏移量,count是查询条数。我们在实现的时候,第一次在查询时score的max也就是当前的时间戳而最小值则为0即可,偏移量也是0,查询条数可自己定义,那么第二次的最大值是上面呢,是第一次查询到数据中的最小值,此时偏移量还是0吗,此时如果是0则上一次查询到的score最小值对应的数据又被查询出来了,那么偏移量是1呢当然也是不行的,如果上一次查询出来的最小值有两个或者多个,执行上述命令查询第二次时,比如第一次按score查score是5 4 4 4 那么第二次查询时是从第一个最小值也就是第一个4开始的,而如果第二次查询时此时偏移量如果为1则 三个4又被重复2查询,那么偏移量如何来定呢,当然是上一次查询分数最小值的重复次数,比如上述例子也就是3,此时第二次查询即可正常进行,那么我们使用代码简单实现一下

5、代码实现


    @Autowired
    private StringRedisTemplate redisTemplate;

    public ReturnModel sub(Long max,Long offset) {
        // 1.获取粉丝id
        Long userId = 1L;

        // 2.查询收件箱
        String key = "FEED:USER:" + userId;
        Set<ZSetOperations.TypedTuple<String>> typedTuples = redisTemplate.opsForZSet()
                .reverseRangeByScoreWithScores(key, 0, max, offset, 3);

        // 3.内容非空判断
        if (typedTuples == null || typedTuples.isEmpty()) {
            return ReturnModel.fail("");
        }
        
        // 4.解析收件箱数据:ids,minTime,offset
        List<Long> ids = new ArrayList<>(typedTuples.size());
        long minTime = 0;         // 下次查询的最大score值
        int os = 0;               // 下次查询的偏移量
        for (ZSetOperations.TypedTuple<String > tuple:typedTuples) {
            // 4.1 获取id
            ids.add(Long.valueOf(tuple.getValue()));
            
            // 4.2 获取分数--时间戳
            long time = tuple.getScore().longValue();
            if (time == minTime) {
                os++;   // 如果是当前最小时间则偏移量加
            } else {    // 不是则重置偏移量,重写定义最小时间
                minTime = time;
                os = 1;
            }
        }

        // 5.根据收件箱id查询对应内容
        // 去数据库查询id对应的内容封装后返回前端

        // 6.返回数据
        // 要将上述准备的下一次查询的偏移量与最小时间戳(作为下次查询的最大score值)一并封装返回
        return ;
    }

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

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

相关文章

Packet Tracer - 在 VTY 线路上配置 ACL

Packet Tracer - 在 VTY 线路上配置 ACL 地址分配表 设备 接口 IP 地址 子网掩码 默认网关 路由器 F0/0 10.0.0.254 255.0.0.0 不适用 PC NIC 10.0.0.1 255.0.0.0 10.0.0.254 笔记本电脑 NIC 10.0.0.2 255.0.0.0 10.0.0.254 拓扑图 目标 第 1 部分&#…

企业性能测试全面解析,一步步教你进行性能测试!

目录 前言&#xff1a; 性能需求调研 性能测试计划制定 性能测试执行 性能测试结果分析与优化 结尾&#xff1a; 前言&#xff1a; 在软件开发过程中&#xff0c;性能测试是一个非常重要的环节。性能测试的主要目的是评估系统在负载情况下的响应时间、吞吐量、稳定性等指…

【C语言】二分查找(含图解)

文章目录 1. 二分查找思想2. 代码实现2.1 未封装函数2.2 封装函数&#xff08;使用while循环&#xff09;2.3 封装函数&#xff08;使用递归&#xff09; 1. 二分查找思想 二分法&#xff1a;二分查找算法是一种在有序数组中查找某一特定元素的搜索算法&#xff0c;其思想就是…

【Linux】多线程操作

文章目录 一. 线程库二. 多线程操作1. 线程创建2. 线程等待3. 线程退出4. 线程取消5. 线程分离6. 返回值 三. 多线程理解结束语 一. 线程库 在Linux下&#xff0c;并没有真正的线程&#xff0c;只有用进程模拟的轻量级进程&#xff0c;所以Linux并没有提供可以直接创建线程的接…

JavaScript教程(三)之 jQuery

JavaScript库 即library&#xff0c;是一个封装好的特定的集合&#xff08;方法和函数&#xff09;。从封装一大堆函数的角度理解库&#xff0c;就是在这个库中&#xff0c;封装了很多预先定义好的函数在里面&#xff0c;比如动画animate、hide、show&#xff0c;比如获取元素…

每日学术速递5.25

CV - 计算机视觉 | ML - 机器学习 | RL - 强化学习 | NLP 自然语言处理 Subjects: cs.CV 1.Chupa: Carving 3D Clothed Humans from Skinned Shape Priors using 2D Diffusion Probabilistic Models 标题&#xff1a;Chupa&#xff1a;使用 2D 扩散概率模型从蒙皮形状先验雕…

理解Window和WindowManager(一)

理解Window和WindowManager(一) Window是一个抽象类,它的具体实现是PhoneWindow,创建一个WindowManager就可以创建一个Window&#xff0c; Window的具体实现位于WindowManagerService中,WindowManager和WindowManagerService是一个IPC过程 为什么使用Window 首先就是Window…

redis持久化【RDB+AOF】持久化双雄

这是redis系列文章之《redis持久化【RDBAOF】持久化双雄》&#xff0c;上一篇文章【redis基础】redis的十大数据类型_努力努力再努力mlx的博客-CSDN博客 感谢大家的支持~ 目录 RDB 什么是RDB RDB的作用 配置文件关于RDB部分 6vs7 操作步骤 修改配置文件&#xff08;本案…

通过python采集整站lazada商品列表数据,支持多站点

要采集整站lazada商品列表数据&#xff0c;需要先了解lazada网站的结构和数据源。Lazada是东南亚最大的电商平台之一&#xff0c;提供各种商品和服务。Lazada的数据源主要分为两种&#xff1a;HTML和API。 方法1&#xff1a;采集HTML数据 步骤1&#xff1a;确定采集目标 首先…

Redis - Redis为什么快

根据官方数据&#xff0c;Redis 的 QPS 可以达到约 100000&#xff08;每秒请求数&#xff09;&#xff0c;有兴趣的可以参考官方的基准程序测试《How fast is Redis&#xff1f;》&#xff0c;官方地址&#xff1a; https://redis.io/topics/benchmarks 横轴是连接数&#xf…

GPT怎样教我用Python进行数据可视化

文章目录 GPT怎样教我用Python进行数据可视化matplotlibpyecharts总结 GPT怎样教我用Python进行数据可视化 &#x1f680;&#x1f680;首先&#xff0c;我们先看一下这段代码&#xff0c;这是我之前写来读取excel文件中xx大学在各个类别中的获奖情况&#xff0c;并保存在一个…

【数据结构】24王道考研笔记——线性表

线性表 目录 线性表定义和基本操作顺序表静态顺序表动态顺序表 链表单链表不带头结点&#xff1a;带头结点&#xff1a; 双链表循环链表循环单链表&#xff1a;循环双链表&#xff1a; 静态链表 顺序表链表比较逻辑结构&#xff1a;存储结构&#xff1a;基本操作&#xff1a; 定…

【JUC基础】11. 并发下的集合类

目录 1、前言 2、并发下的ArrayList 2.1、传统方式 2.1.1、程序正常运行 2.1.2、程序异常 2.1.3、运行期望值不符 2.2、加锁 2.3、synchronizedList 2.4、CopyOnWriteArrayList 3、并发下的HashSet 3.1、CopyOnWriteArraySet 3.2、HashSet底层是什么&#xff1f; 4…

python基础----环境搭建-----01

一 python介绍 1.1 Python 特点 Python 是完全面向对象的语言。函数、模块、数宁、宁符串都是对象&#xff0c;在 Python 中一切皆对象。完全支持继承、重载、多重继承。支持重载运算符&#xff0c;也支持泛型设计。Python 拥有一个强大的标准库&#xff0c;Python 语言的核心…

element-ui菜单el-menu的使用

效果演示 先给大家看一下效果吧 el-menu详解 Menu Attributes# 属性名说明类型可选值默认值mode菜单展示模式stringhorizontal / verticalverticalcollapse是否水平折叠收起菜单&#xff08;仅在 mode 为 vertical 时可用&#xff09;boolean—falseellipsis是否省略多余的子项…

四、 JSP04 Servlet 技术

四、 Servlet 技术 4.1 认识 Servlet Web 容器在处理 JSP 文件时&#xff0c;会将 JSP 文件通过 JSP 容器转换成可识别的 .java 文件 这个 .java 文就是一个 Servlet 类&#xff0c;JSP 技术就是基于 Servlet 实现的 4.1.1 什么是 Servlet Servlet 是一个符合特定规范的 Java…

Linux系统编程学习 NO.5 ——shell命令行的概念以及原理、权限的概念

1.shell命令行的概念以及原理 首先&#xff0c;用户下达指令需求。此时Linux操作系统的内核kernel&#xff0c;并不会直接接收用户下达的指令&#xff0c;因为操作系统不擅长跟用户打交道。那么指令要如何下达呢?这就命令行解释器来对用户的指令进行处理。 1.1.shell命令行的…

每日学术速递5.26

CV - 计算机视觉 | ML - 机器学习 | RL - 强化学习 | NLP 自然语言处理 Subjects: cs.CV 1.Text2NeRF: Text-Driven 3D Scene Generation with Neural Radiance Fields 标题&#xff1a;Text2NeRF&#xff1a;具有神经辐射场的文本驱动 3D 场景生成 作者&#xff1a;Jingb…

从组件化角度聊聊设计工程化

目录 设计系统 设计系统的定义 设计系统的优势 设计系统存在的问题 设计工程化 设计系统探索 设计系统落地实践 Design Token Design Token 实践 设计工程化理想方案构想 展望 参考文献 近几年围绕业务中台化的场景&#xff0c;涌现出了许多低代码平台。面对多组件…

RAW、RGB 、YUV三种图像格式理解

文章目录 1. 背景2. 相关概念2.1 颜色与色彩空间2.2 RAW图像2.3 RGB图像2.4 YUV图像 3. 分类简图 RAW、RGB 、YUV三种图像格式理解 1. 背景 在工作中&#xff0c;经常听到用来描述图像格式的RAW&#xff0c;RGB与YUV&#xff0c;但一直没有系统的进行了解&#xff0c;处于局部认…