当前位置: 首页 > news >正文

瑞吉外卖项目——缓存优化

用户数量多,系统访问量大

频繁访问数据库,系统性能下降,用户体验差

环境搭建

maven坐标

在项目的pom.xml文件中导入spring data redis的maven坐标:

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

在项目的pom.xml文件中导入spring data redis的maven坐标:

配置文件

在项目的application.yml中加入redis相关配置:

spring:redis:host: 172.17.2.94port: 6379password: root@123456database: 0

配置类

在项目中加入配置类RedisConfig:

@Configuration
public class RedisConfig extends CachingConfigurerSupport {@Beanpublic RedisTemplate<Object, Object> redisTemplate(RedisConnectionFactory connectionFactory) {RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();//默认的Key序列化器为:JdkSerializationRedisSerializerredisTemplate.setKeySerializer(new StringRedisSerializer());redisTemplate.setConnectionFactory(connectionFactory);return redisTemplate;}
}

缓存短信验证码

前面我们已经实现了移动端手机验证码登录,随机生成的验证码我们是保存在HttpSession中的。现在需要改造为将验证码缓存在Redis中,具体的实现思路如下:

  1. 在服务端UserController中注入RedisTemplate对象,用于操作Redis

    @Autowired
    private RedisTemplate redisTemplate;
    
  2. 在服务端UserController的sendMsg方法中,将随机生成的验证码缓存到Redis中,并设置有效期为5分钟

    //将生成的验证码缓存到Redis中,并且设置有效期为5分钟
    redisTemplate.opsForValue().set(phone, code, 5, TimeUnit.MINUTES);
    
  3. 在服务端UserController的login方法中,从Redis中获取缓存的验证码,如果登录成功则删除Redis中的验证码

    //从Redis中获取缓存的验证码
    String codeInRedis = (String) redisTemplate.opsForValue().get(phone);
    

缓存菜品数据

前面我们已经实现了移动端菜品查看功能,对应的服务端方法为DishController的list方法,此方法会根据前端提交的查询条件进行数据库查询操作。在高并发的情况下,频繁查询数据库会导致系统性能下降,服务端响应时间增长。现在需要对此方法进行缓存优化,提高系统的性能。
具体的实现思路如下:

  1. 改造DishController的list方法,先从Redis中获取菜品数据,如果有则直接返回,无需查询数据库;如果没有则查询数据库,并将查询到的菜品数据放入Redis。

    List<DishDto> listDto = null;
    //先从redis中获取缓存数据
    String key = "dish_" + dish.getCategoryId() + "_" + dish.getStatus();//dish_1397844263642378242_1
    listDto = (List<DishDto>) redisTemplate.opsForValue().get(key);
    if (listDto != null) {//如果存在,直接返回,无需查询数据库return R.success(listDto);
    }
    //如果不存在,需要查询数据库,将查询到的菜品数据缓存到Redis
    //...查数据库
    redisTemplate.opsForValue().set(key, listDto, 60, TimeUnit.MINUTES);
    
  2. 改造DishController的save和update方法,加入清理缓存的逻辑

    • 方式一
    //清理所有菜品的缓存数据
    Set keys = redisTemplate.keys("dish_*");
    redisTemplate.delete(keys) ;
    
    • 方式二
    //清理某个分类下面的菜品缓存数据
    String key = "dish_" + dishDto.getCategoryId() + "_1";
    redisTemplate.delete(key);
    

在使用缓存过程中,要注意保证数据库中的数据和缓存中的数据一致,如果数据库中的数据发生变化,需要及时清理缓存数据。

Spring Cache

Spring Cache介绍

spring Cache是一个框架,实现了基于注解的缓存功能,只需要简单地加一个注解,就能实现缓存功能。

Spring Cache提供了一层抽象,底层可以切换不同的cache实现。具体就是通过CacheManager接口来统一不同的缓存技术。

CacheManager是Spring提供的各种缓存技术抽象接口。

针对不同的缓存技术需要实现不同的CacheManager:

CacheManager描述
EhCacheCacheManager使用EhCache作为缓存技术
GuavaCacheManager使用Google的Guavacache作为缓存技术
RedisCacheManager使用Redis作为缓存技术

Spring Cache常用注解

注解说明
@EnableCaching开启缓存注解功能
@Cacheable在方法执行前spring先查看缓存中是否有数据,如果有数据,则直接返回缓存数据;
若没有数据,调用方法并将方法返回值放到缓存中
@CachePut将方法的返回值放到缓存中
@CacheEvict将一条或多条数据从缓存中删除
  • 在spring boot项目中,使用缓存技术只需在项目中导入相关缓存技术的依赖包,并在启动类上使用@EnableCaching开启缓存支持即可。
  • 例如,使用Redis作为缓存技术,只需要导入Spring data Redis的maven坐标即可。

CacheEvict(清除缓存)注解支持SpEL(Spring Expression Language,Spring表达式语言),取值时有多种写法:

@CacheEvict(value = "userCache", key = "#pO")//获取方法的第一个参数
//@CacheEvict(value = "userCache ", key = "#root.args[0]")//获取方法的第一个参数
//@CacheEvict(value = "userCache", key = "#id")//获取方法指定名称的参数
//@CacheEvict(value = "userCache", key = "#result")//获取方法的返回值
@DeleteMapping("/{id}")
public void delete (@PathVariable Long id){userService.removeById(id) ;
}

Spring Cache使用方式

在Spring Boot项目中使用Spring Cache的操作步骤(使用redis缓存技术):

  1. 导入maven坐标
    spring-boot-starter-data-redis、spring-boot-starter-cache

  2. 配置application.yml

    spring:cache:redis:time-to-live: 1800000 #设置缓存有效期(ms)
    
  3. 在启动类上加入@EnableCaching注解,开启缓存注解功能

  4. 在Controller的方法上加入@Cacheable、@cacheEvict等注解,进行缓存操作

缓存套餐数据

前面我们已经实现了移动端套餐查看功能,对应的服务端方法为SetmealController的list方法,此方法会根据前端提交的查询条件进行数据库查询操作。

在高并发的情况下,频繁查询数据库会导致系统性能下降,服务端响应时间增长。

现在需要对此方法进行缓存优化,提高系统的性能。

具体的实现思路如下:

  1. 导入Spring Cache和Redis相关maven坐标

    <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId>
    </dependency>
    
    <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId>
    </dependency>
    
  2. 在application.yml中配置缓存数据的过期时间

    spring:cache:redis:time-to-live: 1800000 #设置缓存数据的过期时间(ms)
    
  3. 在启动类上加入@EnableCaching注解,开启缓存注解功能

    @Slf4j//配置log
    @SpringBootApplication
    @ServletComponentScan//开启filter注解扫描
    @EnableTransactionManagement//开启事务注解支持
    @EnableCaching//开启Spring Cache注解方式的缓存功能
    public class ReggieApplication {public static void main(String[] args) {SpringApplication.run(ReggieApplication.class, args);log.info("项目启动成功...");}
    }
    
  4. 在SetmealController的list方法上加入@Cacheable注解

    @GetMapping("/list")
    @Cacheable(value = "setmealCache", key = "#setmeal.categoryId + '_' + #setmeal.status")
    public R<List<Setmeal>> list(Setmeal setmeal)
    
  5. 在SetmealController的save和delete方法上加入CacheEvict注解

    @PostMapping
    @CacheEvict(value = "setmealCache" , allEntries = true)
    public R<String> save(@RequestBody SetmealDto setmealDto)
    
    @DeleteMapping
    @CacheEvict(value = "setmealCache" , allEntries = true)
    public R<String> delete(@RequestParam List<Long> ids)
    

注意:需要为返回的R对象设置序列化器:

public class R<T> implements Serializable
http://www.lryc.cn/news/60641.html

相关文章:

  • 从头创建一个新的浏览器,这合理吗?
  • TypeScript泛型类型和接口
  • docker命令
  • 2023 年 3 月 NFT 月度报告
  • 【http】 get方法和Post方法区别;http和https
  • 第三章 法的渊源与法的分类
  • 在Ubuntu18.04或者20.04下搭建edk2运行环境
  • 多线程编程常用函数用法
  • C++ 标准模板库(Standard Template Library,STL)
  • 一个寄存器的bit2 bit3位由10修改成11,C示例
  • 【洛谷】P1631 序列合并
  • 2023年七大最佳勒索软件解密工具
  • prettier 命令行工具来格式化多个文件
  • 我发现了PMP通关密码!这14页纸直接背!
  • Medical X-rays Dataset汇总(长期更新)
  • 一文告诉你如何做好一份亚马逊商业计划书的框架
  • 原来ChatGPT可以充当这么多角色
  • 数据结构_第十三关(3):归并排序、计数排序
  • “成功学大师”杨涛鸣被抓
  • 【hello C++】内存管理
  • AppArmor零知识学习十二、源码构建(9)
  • Unity - 带耗时 begin ... end 的耗时统计的Log - TSLog
  • Python 智能项目:1~5
  • C++设计模式:面试题精选集
  • 蓝桥 卷“兔”来袭编程竞赛专场-10仿射加密 题解
  • android so库导致的闪退及tombstone分析
  • 图结构基本知识
  • Hibernate 的多种查询方式
  • FreeRTOS 任务调度及相关函数详解(一)
  • 飞桨paddlespeech语音唤醒推理C实现