Redis实现微博好友功能微服务(关注,取关,共同关注)
Posted 共饮一杯无
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Redis实现微博好友功能微服务(关注,取关,共同关注)相关的知识,希望对你有一定的参考价值。
文章目录
需求分析
好友功能是目前社交场景的必备功能之一,一般好友相关的功能包含有:关注/取关、我(他)的关注、我(他)的粉丝、共同关注、我关注的人也关注他等这样一些功能。
类似于这样的功能我们如果采用数据库做的话只是单纯得到用户的一些粉丝或者关注列表的话是很简单也很容易实现, 但是如果我想要查出两个甚至多个用户共同关注了哪些人或者想要查询两个或者多个用户的共同粉丝的话就会很麻烦, 效率也不会很高。但是如果你用redis去做的话就会相当的简单而且效率很高。原因是redis自己本身带有专门针对于这种集合的交集、并集、差集的一些操作。
设计思路
总体思路我们采用mysql + Redis的方式结合完成。MySQL主要是保存落地数据,而利用Redis的Sets数据类型进行集合操作。Sets拥有去重(我们不能多次关注同一用户)功能。一个用户我们存贮两个集合:一个是保存用户关注的人 另一个是保存关注用户的人。
- SADD 添加成员;命令格式:
SADD key member [member …]
----- 关注 - SREM 移除某个成员;命令格式:
SREM key member [member …]
-------取关 - SCARD 统计集合内的成员数;命令格式:
SCARD key
-------关注/粉丝个数 - SISMEMBER 判断是否是集合成员;命令格式:
SISMEMBER key member
---------判断是否关注(如果关注那么只可以点击取关) - SMEMBERS 查询集合内的成员;命令格式:
SMEMBERS key
-------列表使用(关注列表和粉丝列表) - SINTER 查询集合的交集;命令格式:
SINTER key [key …]
--------共同关注、我关注的人关注了他
数据库表设计
这个数据库表的结构比较简单,主要记录了用户id、用户关注的id和关注状态。
CREATE TABLE `t_follow` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`user_id` int(11) DEFAULT NULL COMMENT '当前登录用户的id',
`follow_user_id` int(11) DEFAULT NULL COMMENT '当前登录用户关注的用户的id',
`is_valid` tinyint(1) DEFAULT NULL COMMENT '关注状态,0-没有关注,1-关注了',
`create_date` datetime DEFAULT NULL,
`update_date` datetime DEFAULT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8 ROW_FORMAT=COMPACT COMMENT='用户和用户关注表';
新建好友功能微服务
添加依赖和配置
pom依赖如下:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<parent>
<artifactId>redis-seckill</artifactId>
<groupId>com.zjq</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
<modelVersion>4.0.0</modelVersion>
<artifactId>ms-follow</artifactId>
<dependencies>
<!-- eureka client -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
<!-- spring web -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- mysql -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
<!-- spring data redis -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<!-- mybatis -->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
</dependency>
<!-- commons 公共项目 -->
<dependency>
<groupId>com.zjq</groupId>
<artifactId>commons</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>
<!-- swagger -->
<dependency>
<groupId>com.battcn</groupId>
<artifactId>swagger-spring-boot-starter</artifactId>
</dependency>
</dependencies>
</project>
springboot配置如下:
server:
port: 7004 # 端口
spring:
application:
name: ms-follow # 应用名
# 数据库
datasource:
driver-class-name: com.mysql.cj.jdbc.Driver
username: root
password: root
url: jdbc:mysql://127.0.0.1:3306/seckill?serverTimezone=Asia/Shanghai&characterEncoding=utf8&useUnicode=true&useSSL=false
# Redis
redis:
port: 6379
host: localhost
timeout: 3000
password: 123456
database: 2
# Swagger
swagger:
base-package: com.zjq.follow
title: 好用功能微服务API接口文档
# 配置 Eureka Server 注册中心
eureka:
instance:
prefer-ip-address: true
instance-id: $spring.cloud.client.ip-address:$server.port
client:
service-url:
defaultZone: http://localhost:7000/eureka/
service:
name:
ms-oauth-server: http://ms-oauth2-server/
ms-diners-server: http://ms-users/
mybatis:
configuration:
map-underscore-to-camel-case: true # 开启驼峰映射
logging:
pattern:
console: '%dHH:mm:ss [%thread] %-5level %logger50 - %msg%n'
添加配置类
redis配置类:
package com.zjq.seckill.config;
import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.ClassPathResource;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.data.redis.core.script.DefaultRedisScript;
import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.StringRedisSerializer;
/**
* RedisTemplate配置类
* @author zjq
*/
@Configuration
public class RedisTemplateConfiguration
/**
* redisTemplate 序列化使用的jdkSerializeable, 存储二进制字节码, 所以自定义序列化类
*
* @param redisConnectionFactory
* @return
*/
@Bean
public RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory)
RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();
redisTemplate.setConnectionFactory(redisConnectionFactory);
// 使用Jackson2JsonRedisSerialize 替换默认序列化
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer(Object.class);
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
jackson2JsonRedisSerializer.setObjectMapper(objectMapper);
// 设置key和value的序列化规则
redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
redisTemplate.setKeySerializer(new StringRedisSerializer());
redisTemplate.setHashKeySerializer(new StringRedisSerializer());
redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);
redisTemplate.afterPropertiesSet();
return redisTemplate;
REST配置类:
关注/取关实现
业务逻辑
Mapper实现
Mapper比较简单主要是查询关注信息、添加关注信息、取关或者再次关注。
Service层实现
package com.zjq.seckill.service;
import cn.hutool.core.bean.BeanUtil;
import com.zjq.commons.constant.ApiConstant;
import com.zjq.commons.constant.RedisKeyConstant;
import com.zjq.commons.exception.ParameterException;
import com.zjq.commons.model.domain.ResultInfo;
import com.zjq.commons.model.pojo.Follow;
import com.zjq.commons.model.vo.SignInUserInfo;
import com.zjq.commons.utils.AssertUtil;
import com.zjq.commons.utils.ResultInfoUtil;
import com.zjq.seckill.mapper.FollowMapper;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;
import org.springframework.web.client.RestTemplate;
import javax.annotation.Resource;
import java.util.LinkedHashMap;
/**
* 关注/取关业务逻辑层
* @author zjq
*/
@Service
public class FollowService
@Value("$service.name.ms-oauth-server")
private String oauthServerName;
@Value("$service.name.ms-diners-server")
private String dinersServerName;
@Resource
private RestTemplate restTemplate;
@Resource
private FollowMapper followMapper;
@Resource
private RedisTemplate redisTemplate;
/**
* 关注/取关
*
* @param followUserId 关注的食客ID
* @param isFollowed 是否关注 1=关注 0=取关
* @param accessToken 登录用户token
* @param path 访问地址
* @return
*/
public ResultInfo follow(Integer followUserId, int isFollowed,
String accessToken, String path)
// 是否选择了关注对象
AssertUtil.isTrue(followUserId == null || followUserId < 1,
"请选择要关注的人");
// 获取登录用户信息 (封装方法)
SignInUserInfo dinerInfo = loadSignInDinerInfo(accessToken);
// 获取当前登录用户与需要关注用户的关注信息
Follow follow = followMapper.selectFollow(dinerInfo.getId(), followUserId);
// 如果没有关注信息,且要进行关注操作 -- 添加关注
if (follow == null && isFollowed == 1)
// 添加关注信息
int count = followMapper.save(dinerInfo.getId(), followUserId);
// 添加关注列表到 Redis
if (count == 1)
addToRedisSet(dinerInfo.getId(), followUserId);
return ResultInfoUtil.build(ApiConstant.SUCCESS_CODE,
"关注成功", path, "关注成功");
// 如果有关注信息,且目前处于关注状态,且要进行取关操作 -- 取关关注
if (follow != null && follow.getIsValid() == 1 && isFollowed == 0)
// 取关
int count = followMapper.update(follow.getId(), isFollowed);
// 移除 Redis 关注列表
if (count == 1)
removeFromRedisSet(dinerInfo.getId(), followUserId);
return ResultInfoUtil.build(ApiConstant.SUCCESS_CODE,
"成功取关", path, "成功取关");
// 如果有关注信息,且目前处于取关状态,且要进行关注操作 -- 重新关注
if (follow != null && follow.getIsValid() == 0 && isFollowed == 1)
// 重新关注
int count = followMapper.update(follow.getId(), isFollowed);
// 添加关注列表到 Redis
if (count == 1)
addToRedisSet(dinerInfo.getId(), followUserId);
return ResultInfoUtil.build(ApiConstant.SUCCESS_CODE,
"关注成功", path, "关注成功");
return ResultInfoUtil.buildSuccess(path, "操作成功");
/**
* 添加关注列表到 Redis
*
* @param dinerId
* @param followUserId
*/
private void addToRedisSet(Integer dinerId, Integer followUserId)
redisTemplate.opsForSet().add(RedisKeyConstant.following.getKey() + dinerId, followUserId);
redisTemplate.opsForSet().add(RedisKeyConstant.followers.getKey() + followUserId, dinerId);
/**
* 移除 Redis 关注列表
*
* @param dinerId
* @param followUserId
*/
private void removeFromRedisSet(Integer dinerId, Integer followUserId)
redisTemplate.opsForSet().remove(RedisKeyConstant.following.getKey() + dinerId, followUserId);
redisTemplate.opsForSet().remove(RedisKeyConstant.followers.getKey() + followUserId, dinerId);
/**
* 获取登录用户信息
*
* @param accessToken
* @return
*/
private SignInUserInfo loadSignInDinerInfo(String accessToken)
// 必须登录
AssertUtil.mustLogin(accessToken);
String url = oauthServerName + "user/me?access_token=accessToken";
ResultInfo resultInfo = restTemplate.getForObject(url, ResultInfo.class, accessToken);
if (resultInfo.getCode() != ApiConstant.SUCCESS_CODE)
throw new ParameterException(resultInfo.getMessage());
SignInUserInfo dinerInfo = BeanUtil.fillBeanWithMap((LinkedHashMap) resultInfo.getData(),
new SignInUserInfo(), false);
return dinerInfo;
Controller实现
package com.zjq.seckill.controller;
import com.zjq.commons.model.domain.ResultInfo;
import com.zjq.seckill.service.FollowService;
import 以上是关于Redis实现微博好友功能微服务(关注,取关,共同关注)的主要内容,如果未能解决你的问题,请参考以下文章