社区系统项目复盘-4

Redis高性能存储

使用redis实现了点赞、关注相关的功能,优化了登录模块。

什么是Redis?

Redis是一个开源的key-value存储系统,支持多种数据类型,包括string(字符串),list(链表),set(集合),zset(有序集合)和hash(哈希)。Redis操作都是原子性的(有一个失败则都失败)。Redis支持各种方式的排序。Redis数据可以缓存在内存中,也可以周期性的写入到磁盘中(即持久化操作),在此基础上实现了主从(master-slave)同步。

为什么使用Redis?

Redis将所有的数据都存放在内存中,所以读写速度很快。同时,Redis将内存中的数据以快照或日志的形式保存到硬盘上,以保证数据的安全性。

Spring是怎么整合Redis的?

第一步:引入依赖

  • spring-boot-starter-data-redis
<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

第二步:对redis进行配置

  • 配置数据库参数
# RedisProperties
spring.redis.database=11
spring.redis.host=localhost
spring.redis.port=6379
  • 编写配置类,构造Spring Redis的核心组件RedisTemplate,这个组件是由Spring提供的
@Configuration
public class RedisConfig {
    @Bean
    public RedisTemplate<String,Object> redisTemplate(RedisConnectionFactory factory){
        RedisTemplate<String,Object> template = new RedisTemplate<>();
        template.setConnectionFactory(factory);

        // 设置key的序列化方式
        template.setKeySerializer(RedisSerializer.string());
        // 设置value的序列化方式
        template.setValueSerializer(RedisSerializer.json());
        // 设置hash的key的序列化方式
        template.setHashKeySerializer(RedisSerializer.string());
        // 设置hash的value的序列化方式
        template.setHashValueSerializer(RedisSerializer.json());

        template.afterPropertiesSet();
        return template;
    }
}

第三步:访问redis的的不同数据类型

  • redisTemplate.opForValue()
  • redisTemplate.opForHash()
  • redisTemplate.opForList()
  • redisTemplate.opForSet()
  • redisTemplate.opForZset()

Redis事务管理

redis也是一个数据库,它也是支持事务的,redis事务管理的机制:当启用事务以后,再执行redis命令的时候,并不会立马执行命令,而是将命令放到一个队列里先存着,直到提交事务的时候才会将队列中的命令一次性发送给redis服务器一起执行。所以,redis使用事务的时候,不要在事务过程中做查询操作,因为查询不会立马显示结果,查询操作应当放在事务前或者事务后。

因为声明式事务只可以精确到方法,所以一般使用编程式事务,编程式事务示例如下:

// 编程式事务
    @Test
    public void testTransactional(){
        Object obj = redisTemplate.execute(new SessionCallback() {
            @Override
            public Object execute(RedisOperations operations) throws DataAccessException {
                String redisKey = "test:tx";
                operations.multi();  // 启用事务

                operations.opsForSet().add(redisKey,"zhangsan");
                operations.opsForSet().add(redisKey,"lisi");
                operations.opsForSet().add(redisKey,"wangwu");
				  // 查询时是没有结果的。所以redis在管理事务的时候中间不要做查询,无效
                System.out.println(operations.opsForSet().members(redisKey));
                return operations.exec();  // 提交事务
            }
        });
        System.out.println(obj);
    }

点赞

点赞部分主要有两大块功能,第一块就是点赞,第二块就是收到的赞

主要显示页面:首页的帖子列表部分显示帖子的赞的数量;帖子详情上需要显示赞的数量以及点赞状态;个人主页上需要显示收到的赞的数量。

点赞:支持对帖子、评论点赞,第1次点赞,第2次取消点赞;需要显示点赞数量:对于首页点赞数量:统计帖子的点赞数量;对于详情页点赞数量:统计点赞数量,显示点赞状态。

详细步骤:点赞,首先调用点赞业务,查询点赞数量,当前点赞状态,~~根据点赞状态确定是否出发点赞事件(后面系统通知部分的功能),重新计算帖子分数(后面热帖的功能),~~最后返回响应。

注意⚠️:点赞部分的存储使用redis的set来实现,将点赞者的用户ID存到set中。

收到的赞:需要重构点赞功能,以用户为key,记录点赞数量,增加点赞数量使用increment(key),减少点赞数量使用decrement(key);开发个人主页,以用户为key,显示用户收到的点赞的数量。

详细步骤:业务层重构点赞方法,另外需要添加一个查询某个用户获得的赞的数量的业务层方法。开发个人主页时,需要显示的内容主要有:用户信息,获得点赞数量,关注数量,粉丝数量,是否已关注,下一部分实现关注相关功能。

注意⚠️:使用了redis事务管理,把点赞与记录点赞数量两个redis命令放在一个事务。
收获的赞的数量使用redis的String来存储,存储的值为整数类型。

在这里插入图片描述

关注

关注、取消关注

  • 需求:开发关注、取消关注功能;统计用户的关注数、粉丝数。实现个人首页显示关注数量,粉丝数量,是否已关注的功能。
  • 关键:若A关注了B,则A是B的Follower(粉丝),B是A的Followee(目标);关注的用户可以是用户、帖子、题目等,在实现时将这些目标抽象为实体【这一句比较重要,抽象化】

⚠️注意:

  • 关注、取关功能,也是需要使用redis事务管理的,因为需要同时更新存储关注的rediskey和存储粉丝的rediskey。
  • 对于关注或者粉丝的存储,采用的是redis的zset数据结构,score使用的是当前系统时间。
  • 关注、取关功能为异步请求

关注列表、粉丝列表

  • 需求:在个人主页,点击关注或者粉丝,会相应的显示关注列表和粉丝列表,分页显示。
  • 关键:
    • 业务层:查询某个用户关注的人,支持分页;查询某个用户的粉丝,支持分页。
    • 表现层:处理“查询关注的人”,“查询粉丝”请求。

⚠️注意:业务层查询的信息用map来存,因为列表中不只需要有用户的信息,还需要有关注的时间。主要步骤是:先通过Redis查出来userid集合,然后遍历该集合,从根据userid从user数据库中查用户信息;关注时间是通过Redis来查询的,score就是关注时间。

在这里插入图片描述

优化登录模块

主要使用Redis优化3个方面:存储验证码,存储用户登录凭证,缓存用户信息

为什么要使用redis进行优化呢?

  • 使用Redis存储验证码
    • 验证码需要频繁的访问与刷新,对性能要求比较高
    • 验证码不需要永久保存,通常在很短的时间后就会失效
    • 分布式部署时,存在Session共享的问题
  • 使用Redis存储登录凭证
    • 处理每次请求时,都要查询用户的登录凭证,访问的频率非常高
  • 使用Redis缓存用户信息
    • 处理每次请求时,都要根据凭证查询用户信息,访问的频率非常高

在这里插入图片描述

⚠️注意:

  • 生成验证码的时候,用户还没有登录,此时无法获得userid,无法通过userid构造验证码对应的rediskey。所以生成验证码时临时给客户端颁发一个代表验证码归属的字符串就可以,这个字符串使用cookie发给客户端,同时使用该字符串构造存储验证码的rediskey。
// 验证码的归属
String kaptchaOwner = CommunityUtil.generateUUID();
Cookie cookie = new Cookie("kaptchaOwner",kaptchaOwner);
cookie.setMaxAge(60);
cookie.setPath(contextPath);
response.addCookie(cookie);
// 将验证码存入Redis
String redisKey = RedisKeyUtil.getKaptchaKey(kaptchaOwner);
redisTemplate.opsForValue().set(redisKey,text,60, TimeUnit.SECONDS);
  • 登录时检查验证码,此时标识验证码归属的字符串通过cookie传过来,然后我们通过这个字符串从redis里获得验证码的正确值,然后再校验验证码是否正确。
//检查验证码
// String kaptcha = (String) session.getAttribute("kaptcha");
String kaptcha = null;
if(StringUtils.isNoneBlank(kaptchaOwner)){
	String redisKey = RedisKeyUtil.getKaptchaKey(kaptchaOwner);
	kaptcha = (String) redisTemplate.opsForValue().get(redisKey);
}
if(StringUtils.isBlank(kaptcha) || StringUtils.isBlank(code) || !kaptcha.equalsIgnoreCase(code)){
	model.addAttribute("codeMsg","验证码不正确!");
	return "/site/login";
}
  • 使用redis存取登录凭证,首先使用@Deprecated注解废弃LoginTicketMapper,修改登录时存储登录凭证的方式,修改退出登录时修改登录凭证状态的方式,修改查询登录凭证的方式。之前是将登录凭证存储在mysql数据库表中,现在把它存储到redis中里。
  • 查询user时,先尝试从缓存中取值,如果有,则直接返回值,如果没有,就从数据库中查找,将查找结果存到缓存中。数据变更时(激活的时候,修改密码的时候,上传头像的时候),需要更新数据库,这时候为了保持缓存和数据库中信息的一致性,可以有两种方法:一是同步更新缓存,二是直接将缓存中数据删掉,本项目采取方法二。
// 1.优先从缓存中取值
    private User getCache(int userId){
        String redisKey = RedisKeyUtil.getUserKey(userId);
        return (User) redisTemplate.opsForValue().get(redisKey);
    }
    // 2.取不到时初始化缓存数据
    private User initCache(int userId){
        User user = userMapper.selectById(userId);
        String redisKey = RedisKeyUtil.getUserKey(userId);
        redisTemplate.opsForValue().set(redisKey,user,3600, TimeUnit.SECONDS);
        return user;
    }
    // 3.数据变更时清除缓存数据
    private void clearCache(int userId){
        String redisKey = RedisKeyUtil.getUserKey(userId);
        redisTemplate.delete(redisKey);
    }

小结:点赞部分用了Set数据结构,点赞数量用的String数据结构。关注用的Zset数据结构。验证码的存取用的String数据结构。登录凭证的存取用的String数据结构。缓存用户信息用的String数据结构。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
IT项目复盘会议模板用于总结和评估已完成的IT项目,以确定项目的成功与短板,并为未来的项目提供改进和学习的机会。以下是一个常见的IT项目复盘会议模板的简要内容: 1. 会议目的: - 确定项目成功的关键因素和挑战 - 分享项目经验和教训 - 提供改进建议和措施 2. 项目概述: - 提供项目背景和目标 - 明确项目范围和时间表 3. 成功因素: - 总结项目的成功因素,例如团队合作、领导支持、资源管理等 - 强调成功因素的积极影响和实施方法 4. 挑战和教训: - 讨论项目中的挑战和困难 - 反思潜在的教训和应对方法,包括项目管理、沟通、风险管理等方面 5. 项目绩效评估: - 评估项目的时间表、预算和质量目标的达成情况 - 分析任何超出或不足的情况,并确定其中的原因 6. 改进建议: - 提供项目改进的建议和措施 - 确定未来项目中需要避免的问题,并提出解决方案 7. 团队反馈: - 鼓励团队成员分享对项目的观点和经验 - 倾听并考虑团队成员提出的意见和建议 8. 行动计划: - 制定行动计划,明确改进措施和责任人 - 确定实施时间表和监控方法 9. 会议总结: - 概括会议讨论的要点和重点 - 强调重要的改进措施和下一步行动 IT项目复盘会议模板的内容可以根据实际项目的需要进行调整和添加,以确保全面而有针对性的评估和总结。该模板有助于促进项目范围、进度和质量的改进,提高项目的成功率和效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值