苍穹外卖day07缓存部分分析

苍穹外卖Day07部分聚焦于缓存功能的实现与优化,通过引入redis缓存机制,结合Spring Cache 注解,降低了数据库负载,提升其响应速度。

以下是清除缓存功能代码:

java 复制代码
@RestController
@RequestMapping("/admin/dish")
@Slf4j
@Api("菜品相关接口")
public class DishController {

    @Autowired
    DishService dishService;

    @Autowired
    RedisTemplate redisTemplate;

    @GetMapping("/list")
    @ApiOperation("根据分类id查询菜品")
    public Result<List<DishVO>> getByCategoryId(Long categoryId) {
        log.info("根据分类id查询菜品");
        String key = "dish:" + categoryId;
        List<DishVO> a = (List<DishVO>) redisTemplate.opsForValue().get(key);
        System.out.println(a);
        if (a != null && a.size() > 0) {
            return Result.success(a);
        }
        Dish dish = new Dish();
        dish.setCategoryId(categoryId);
        dish.setStatus(StatusConstant.DISABLE);
        a = dishService.listWithFlavor(dish);
        redisTemplate.opsForValue().set(key, a);
        return Result.success(a);
    }

    @PostMapping("/status/{status}")
    @ApiOperation("菜品起售停售")
    public Result startOrStop(@PathVariable Integer status, Long id) {
        dishService.startOrStop(status,id);
        cleanCache();
        return Result.success();
    }
    private void cleanCache() {
        Set set = redisTemplate.keys("dish:*");
        redisTemplate.delete(set);
    }
}

在startOrStop方法中,当执行菜品起售停售操作后,会调用 cleanCache 私有方法来清理 Redis 缓存。在 cleanCache 方法里,通过 Set set = redisTemplate.keys("dish:*"); 利用 Redis 的 keys 命令获取所有以 "dish:" 开头的键值,也就是获取所有与菜品相关的缓存键,然后使用 redisTemplate.delete(set),将这些键对应的缓存数据全部删除。

这样可以确保在菜品状态发生改变后,缓存中的菜品数据能及时更新,避免前端获取到旧的不符合实际状态的菜品信息,从而保证了缓存数据与数据库数据的一致性。但是使用keys命令在生产环境中如果数据量很大可能会影响性能,因为它需要遍历所有键,可以考虑采用更精准的缓存失效策略,比如根据具体变更的菜品 id 来有针对性地删除相关缓存,而不是批量删除所有菜品缓存,也可以使用Spring Cache注解。

Spring Cache

  1. @EnableCaching:加在启动类上,用于开启缓存注解功能,使得项目中可以使用Spring Cache的其他注解。
  2. @Cacheable:放在方法上,在方法执行前先查询缓存中是否存在缓存数据,存在数据直接将数据返回;没有缓存数据,通过反射调用方法并将方法的返回值放到缓存中。
  3. @CachePut:将方法的返回值放到缓存中,通常用于在方法执行完毕后更新缓存中的数据。
  4. @CacheEvict:用于将一条或多条数据从缓存中删除,可以根据具体的 key 删除指定的缓存数据,也可以使用allEntries = true 删除整个缓存名称下的所有数据。

以下为使用spring cahce注解的代码:

java 复制代码
@RestController
@RequestMapping("/admin/dish")
@Slf4j
@Api("菜品相关接口")
@EnableCaching
public class DishController {

    @Autowired
    DishService dishService;

    @Autowired
    RedisTemplate redisTemplate;

    @GetMapping("/list")
    @ApiOperation("根据分类id查询菜品")
    @Cacheable(cacheNames = "dishCache", key = "#categoryId")
    public Result<List<DishVO>> getByCategoryId(Long categoryId) {
        log.info("根据分类id查询菜品");
        Dish dish = new Dish();
        dish.setCategoryId(categoryId);
        dish.setStatus(StatusConstant.DISABLE);
        List<DishVO> a = dishService.listWithFlavor(dish);
        return Result.success(a);
    }

    @PostMapping("/status/{status}")
    @ApiOperation("菜品起售停售")
    @CacheEvict(cacheNames = "dishCache", allEntries = true)
    public Result startOrStop(@PathVariable Integer status, Long id) {
        dishService.startOrStop(status, id);
        return Result.success();
    }
}

这段代码使用@Cacheable(cacheNames = "dishCache", key = "#categoryId") 注解, cacheNames 指定了缓存的名称空间为 "dishCache",用于将菜品相关的缓存数据统一归类管理,方便后续维护与排查问题。

key = "#categoryId" 则以传入的菜品分类 id作为缓存的键,使得在查询菜品时,系统能够依据分类 id 精准地在 "dishCache" 缓存区域中查找对应数据。当缓存中有匹配的数据时,直接返回缓存数据,不再执行方法体中的数据库查询代码,大大加快了响应速度。

相关推荐
烤麻辣烫5 分钟前
黑马程序员苍穹外卖(新手) DAY3
java·开发语言·spring boot·学习·intellij-idea
妮妮喔妮10 分钟前
JAVA反射的介绍(优缺点)
java·开发语言
i***486115 分钟前
Redis重大版本整理(Redis2.6-Redis7.0)
java·数据库·redis
YQ_ZJH23 分钟前
Redisson 看门狗机制详解
java·redis
那我掉的头发算什么27 分钟前
【javaEE】多线程——线程安全进阶☆☆☆
java·jvm·安全·java-ee·intellij-idea
悟空CRM服务35 分钟前
我用一条命令部署了完整CRM系统!
java·人工智能·开源·开源软件
组合缺一36 分钟前
Solon AI 开发学习 - 1导引
java·人工智能·学习·ai·openai·solon
百***490038 分钟前
基于SpringBoot和PostGIS的各省与地级市空间距离分析
java·spring boot·spring
百***266339 分钟前
使用 Logback 的最佳实践:`logback.xml` 与 `logback-spring.xml` 的区别与用法
xml·spring·logback
电摇小人1 小时前
科学备赛今年NOIP!!
java·开发语言