缓存菜品-03.代码开发-清理缓存数据

一.问题分析

我们首先将馋嘴牛蛙的价格修改成66。然后我们看一下小程序端的价格。

缓存菜品-03.代码开发-清理缓存数据_第1张图片

还是88。这是因为我们的数据并没有同步。

缓存菜品-03.代码开发-清理缓存数据_第2张图片

我们小程序端是从redis查询出来的,而我们修改管理端实际上是直接修改了数据库,因此会造成数据的不一致。

那么我们在哪些情况下需要清理redis中的缓存数据呢?

1.修改操作,修改后redis中的数据就和新的不同,需要将其删除掉。

2.删除菜品,当菜品被删除后,小程序端不能展示出来。

3. 启售/停售菜品,启售/停售都会影响小程序端的菜品展示,因此也要将redis中的数据清除。

4.新增菜品也要清除,因为新增也会使得redis和sql数据库不一致。

因为用户不能对redis数据库进行操作,因此我们要在admin,即管理端对redis中的数据进行内存清除。

package com.sky.controller.admin;

import com.sky.dto.DishDTO;
import com.sky.dto.DishPageQueryDTO;
import com.sky.entity.Dish;
import com.sky.result.PageResult;
import com.sky.result.Result;
import com.sky.service.DishService;
import com.sky.vo.DishVO;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.*;

import java.util.HashSet;
import java.util.List;
import java.util.Set;

@RestController
@RequestMapping("/admin/dish")
@Api(tags = "菜品相关接口")
@Slf4j
public class DishController {
    @Autowired
    private DishService dishService;

    @Autowired
    private RedisTemplate redisTemplate;


    /**
     * 新增菜品
     * @param dishDTO
     * @return
     */
    @PostMapping
    @ApiOperation("新增菜品")
    public Result save(@RequestBody DishDTO dishDTO) {  // 前端传递过来的Json格式的数据,要使用@RequestBody注解修饰变量来接收
        log.info("新增菜品:{}",dishDTO);
        dishService.saveWithFlavor(dishDTO);
        cleanCache("dish_" + dishDTO.getCategoryId());    // sql数据库当中的数据发生了修改,要将redis中的旧数据进行清除
        return Result.success();
    }

    /**
     * 菜品分页查询
     * @param dishPageQueryDTO
     * @return
     */
    @GetMapping("/page")
    @ApiOperation("菜品分页查询")
    public Result page(DishPageQueryDTO dishPageQueryDTO) {
        log.info("菜品分页查询:{}",dishPageQueryDTO);
        PageResult pageResult = dishService.pageQuery(dishPageQueryDTO);
        return Result.success(pageResult);
    }

    /**
     * 批量删除菜品
     * @param ids
     * @return
     */
    @DeleteMapping
    @ApiOperation("批量删除菜品")
    public Result deleteBatch(@RequestParam List ids) {
        log.info("批量删除菜品:{}",ids);
        dishService.deleteByIds(ids);
        /*
        精确清理redis中的缓存
        Set categoryIds = dishService.getCategoryIdsByIds(ids);      // 通过查找被删除菜品所属的分类id集合,将redis中该分类集合所属的值全部删除
        Set dish_categoryIds = new HashSet<>();
        for (Long categoryId : categoryIds) {
            dish_categoryIds.add("dish_" + categoryId);
        }
        redisTemplate.delete(dish_categoryIds);
         */
        cleanCache("dish_*");
        return Result.success();
    }

    /**
     * 根据id查询菜品
     * @param id
     * @return
     */
    @GetMapping("/{id}")
    @ApiOperation("根据id查询菜品")
    public Result getById(@PathVariable Long id) {
        log.info("根据id查询菜品:{}",id);
        DishVO dishVO = dishService.getByIdWithFlavor(id);
        return Result.success(dishVO);
    }

    /**
     * 修改菜品
     * @param dishDTO
     * @return
     */
    @PutMapping()
    @ApiOperation("修改菜品")
    public Result update(@RequestBody DishDTO dishDTO) {
        log.info("修改菜品:{}",dishDTO);
        dishService.updateWithFlavor(dishDTO);
        cleanCache("dish_*");
        return Result.success();
    }

    /**
     * 菜品起售、停售
     * @param status
     * @return
     */
    @PostMapping("/status/{status}")
    @ApiOperation("菜品起售、停售")
    public Result startOrStop(@PathVariable Integer status, Long id) {
        log.info("菜品起售、停售");
        dishService.startOrStop(status,id);
        cleanCache("dish_*");
        return Result.success();
    }


    /**
     * 根据分类id查询菜品,该接口仅仅是为了在用户端根据分类id显示出对应的菜品时,能够将其在售的菜品显示出来,对于管理端本身没用
     * @param categoryId
     * @return
     */
    @GetMapping("/list")
    @ApiOperation("根据分类id查询菜品")
    public Result> list(Long categoryId) {
        List list = dishService.getByCategoryId(categoryId);
        return Result.success(list);
    }

    /**
     * 清除redis中的缓存数据
     * @param pattern
     */
    private void cleanCache(String pattern) {
        // 模糊清理:只要菜品发生变动就将redis中的所有缓存清理
        Set keys = redisTemplate.keys(pattern);    //  查询出redis中所有以"dish_"开头的key,并将其封装在集合中
        redisTemplate.delete(keys);     // 清除redis中的菜品缓存
    }
}

在新增菜品清理缓存数据时,因为分类id很容易获取,因此key很容易构建并查询。所以在新增菜品清理缓存数据时我们可以进行精准清理,即哪个分类修改了就清除哪个分类下的redis数据。

其他的操作修改我们直接将所有的redis中的"dish_"开头的键所对应的数据全部清除。

可以统一抽取清理缓存的方法。

/**
     * 清除redis中的缓存数据
     * @param pattern
     */
    private void cleanCache(String pattern) {
        // 模糊清理:只要菜品发生变动就将redis中的所有缓存清理
        Set keys = redisTemplate.keys(pattern);    //  查询出redis中所有以"dish_"开头的key,并将其封装在集合中
        redisTemplate.delete(keys);     // 清除redis中的菜品缓存
    }

你可能感兴趣的:(苍穹外卖,缓存,spring,springboot,苍穹外卖,微信,微信小程序)