重学SpringBoot3-集成Redis(十二)之点赞功能实现

更多SpringBoot3内容请关注我的专栏:《SpringBoot3》

期待您的点赞👍收藏⭐评论✍

重学SpringBoot3-集成Redis(十二)之点赞功能实现

  • [1. 点赞功能的场景分析](#1. 点赞功能的场景分析)
  • [2. 项目环境配置](#2. 项目环境配置)
    • [2.1. 依赖引入](#2.1. 依赖引入)
    • [2.2. Redis 配置](#2.2. Redis 配置)
  • [3. 点赞功能的实现](#3. 点赞功能的实现)
  • [4. 点赞功能的详细解释](#4. 点赞功能的详细解释)
    • [4.1. 用户点赞和取消点赞](#4.1. 用户点赞和取消点赞)
    • [4.2. 统计点赞数](#4.2. 统计点赞数)
    • [4.3. 判断用户是否点赞](#4.3. 判断用户是否点赞)
  • [5. Redis Set 数据结构的优势](#5. Redis Set 数据结构的优势)
  • [6. 总结](#6. 总结)

在现代的应用中,点赞功能 是一个非常常见的需求,尤其在社交媒体、博客等平台上。Redis 作为一个高性能的键值存储系统,由于其读写速度快支持丰富的数据结构 ,因此非常适合用来实现实时的点赞功能。本文将介绍如何结合 Spring Boot 3Redis 来实现一个高效的点赞功能。


1. 点赞功能的场景分析

点赞功能通常涉及以下场景:

  • 用户点赞或取消点赞:某个用户对某篇文章或某条评论进行点赞或取消操作。
  • 统计点赞数量:实时显示某个对象(如文章、视频、评论)的总点赞数。
  • 用户点赞状态查询:判断某个用户是否对某个对象点赞过。

Redis 通过Set 数据结构可以很好地解决这些问题。Redis 的 Set 不允许重复元素,且支持快速添加、删除、判断成员是否存在等操作。


2. 项目环境配置

2.1. 依赖引入

首先,在 pom.xml 中引入 Spring Boot 3Redis 的相关依赖,具体参考重学SpringBoot3-集成Redis(一)之基本使用

xml 复制代码
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

2.2. Redis 配置

application.yml 中配置 Redis 连接:

yaml 复制代码
spring:
  data:
    redis:
      host: localhost
      port: 6379            # Redis 端口
      password: redis123456 # 如果有密码可以在这里配置
      lettuce:
        pool:
          max-active: 100    # 最大并发连接数
          max-idle: 50       # 最大空闲连接数
          min-idle: 10       # 最小空闲连接数

3. 点赞功能的实现

在实现点赞功能时,通常会用 Redis 的 Set 数据结构来存储每个对象(如文章、视频)的点赞用户列表。每次点赞操作就是往这个 Set 中添加用户 ID,取消点赞则是从 Set 中移除用户 ID。

3.1. 点赞服务层

java 复制代码
package com.coderjia.boot310redis.service;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Service;

/**
 * @author CoderJia
 * @create 2024/10/10 下午 09:45
 * @Description
 **/
@Service
public class LikeService {

    @Autowired
    private StringRedisTemplate redisTemplate;

    // Redis Key 前缀
    private static final String LIKE_KEY_PREFIX = "like:";  

    // 用户点赞
    public void likePost(String postId, String userId) {
        String redisKey = LIKE_KEY_PREFIX + postId;
        redisTemplate.opsForSet().add(redisKey, userId);
    }

    // 用户取消点赞
    public void unlikePost(String postId, String userId) {
        String redisKey = LIKE_KEY_PREFIX + postId;
        redisTemplate.opsForSet().remove(redisKey, userId);
    }

    // 查询某个帖子点赞数
    public Long getLikeCount(String postId) {
        String redisKey = LIKE_KEY_PREFIX + postId;
        return redisTemplate.opsForSet().size(redisKey);
    }

    // 判断用户是否点赞
    public boolean hasLiked(String postId, String userId) {
        String redisKey = LIKE_KEY_PREFIX + postId;
        return Boolean.TRUE.equals(redisTemplate.opsForSet().isMember(redisKey, userId));
    }
}

3.2. 点赞控制器

java 复制代码
package com.coderjia.boot310redis.demos.web;

import com.coderjia.boot310redis.service.LikeService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;

/**
 * @author CoderJia
 * @create 2024/10/10 下午 09:46
 * @Description
 **/
@RestController
@RequestMapping("/like")
public class LikeController {

    @Autowired
    private LikeService likeService;

    // 点赞操作
    @PostMapping("/like")
    public String like(@RequestParam String postId, @RequestParam String userId) {
        likeService.likePost(postId, userId);
        return "Liked post: " + postId + " by user: " + userId;
    }

    // 取消点赞
    @PostMapping("/unlike")
    public String unlike(@RequestParam String postId, @RequestParam String userId) {
        likeService.unlikePost(postId, userId);
        return "Unliked post: " + postId + " by user: " + userId;
    }

    // 获取点赞数
    @GetMapping("/count")
    public Long getLikeCount(@RequestParam String postId) {
        return likeService.getLikeCount(postId);
    }

    // 检查用户是否点赞
    @GetMapping("/status")
    public boolean hasLiked(@RequestParam String postId, @RequestParam String userId) {
        return likeService.hasLiked(postId, userId);
    }
}

3.3. 演示

点赞操作

curl 复制代码
curl -X POST http://localhost:8080/like/like?postId=p101&userId=1

获取点赞数

curl 复制代码
curl http://localhost:8080/like/count?postId=p101

检查用户是否点赞

curl 复制代码
curl http://localhost:8080/like/status?postId=p101&userId=1

取消点赞

curl 复制代码
curl -X POST http://localhost:8080/like/unlike?postId=p101&userId=1

4. 点赞功能的详细解释

4.1. 用户点赞和取消点赞

每当用户点赞时,我们将用户 ID 存入 Redis 的 Set 中。由于 Redis 的 Set 不允许重复元素,用户多次点赞同一篇文章也只会被记录一次。

java 复制代码
redisTemplate.opsForSet().add(redisKey, userId);

取消点赞则是将用户 ID 从 Set 中移除:

java 复制代码
redisTemplate.opsForSet().remove(redisKey, userId);

4.2. 统计点赞数

统计点赞数非常简单,直接调用 Redis 的 size() 方法即可:

java 复制代码
redisTemplate.opsForSet().size(redisKey);

这比使用传统数据库查询要快得多,尤其在大量用户点赞的情况下,Redis 能保持高性能。

4.3. 判断用户是否点赞

可以通过 Redis 的 isMember() 方法来判断某个用户是否已经对某篇文章点赞:

java 复制代码
redisTemplate.opsForSet().isMember(redisKey, userId);

这一点对前端显示用户是否已点赞的状态非常重要,用户体验更好。


5. Redis Set 数据结构的优势

Redis 的 Set 数据结构非常适合用来存储点赞功能的用户列表,原因有以下几点:

  1. 唯一性:Redis Set 不允许重复元素,确保用户对同一篇文章只能点赞一次。
  2. 高性能:Redis 是内存级存储,读写速度极快,适合大规模的点赞操作。
  3. 丰富的操作:Set 提供了丰富的操作,如添加成员、删除成员、计算数量、检查成员是否存在等,这些操作都是 O(1) 复杂度,性能非常高。

6. 总结

通过结合 Spring Boot 3Redis,我们可以轻松实现高效的点赞功能,并利用 Redis 的 Set 数据结构实现去重、快速统计等操作。相比于传统的数据库操作,使用 Redis 实现的点赞功能性能更高、扩展性更好,尤其适合用户量大、点赞操作频繁的应用场景。

持久化到数据库

仅使用 Redis:适用于对数据一致性要求不高的场景,比如短期有效的点赞数据,或者系统对少量点赞数据丢失不敏感。

Redis + MySQL方案 :适用于对数据一致性要求高的场景,比如电商、社交平台中,点赞数据不能丢失,且需要长期保存。通常采用异步持久化,具体流程可以是:

  1. 用户点赞时,首先将数据写入 Redis。
  2. 通过定时任务(如每隔几分钟)或消息队列,将 Redis 中的点赞数据同步到数据库。
  3. 在定时任务或队列消费过程中,可以批量将点赞数据写入数据库,降低数据库的写入压力。
相关推荐
咖猫7 分钟前
Google guava 最佳实践 学习指南之08 `BiMap`(双向映射)
java·开发语言·guava
娶个名字趴16 分钟前
Redis(2)常用命令
java·数据库·redis·缓存
滿17 分钟前
处理错误的两种方式:try...catch 与 then...catch
java·开发语言
Ttang2330 分钟前
Tomcat原理(4)——尝试手动Servlet的实现
java·开发语言·servlet·java-ee·tomcat·intellij-idea
lzz的编码时刻1 小时前
Java 8 Optional 详细使用教程-优雅解决NPE
java
计算机毕设指导61 小时前
基于Springboot林业产品推荐系统【附源码】
java·开发语言·spring boot·后端·mysql·spring·intellij-idea
CQU_JIAKE1 小时前
12.8&12.9[java exp4][debug]跨域问题原因详解
java·开发语言
南宫生2 小时前
力扣-图论-15【算法学习day.65】
java·学习·算法·leetcode·图论
cloud___fly2 小时前
黑马Redis数据结构学习笔记
数据结构·redis·笔记·学习
洛嘚2 小时前
@FeignClient用于Nacos微服务间的接口调用
java·服务器