瑞吉外卖(优化)——缓存菜品数据

实现思路

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

  • 改造list方法,先从redis中读取数据,如果有则直接返回;如果redis中没有数据,则进行数据库的查询,并将查询到的菜品数据放入redis。
  • 改造DishController中的save和update方法,加入清理缓存的逻辑。

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

代码如下

package com.jie.reggie.controller;

import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
import com.baomidou.mybatisplus.core.conditions.update.UpdateWrapper;
import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
import com.jie.reggie.common.R;
import com.jie.reggie.dto.DishDto;
import com.jie.reggie.entity.Category;
import com.jie.reggie.entity.Dish;
import com.jie.reggie.entity.DishFlavor;
import com.jie.reggie.service.CategoryService;
import com.jie.reggie.service.DishFlavorService;
import com.jie.reggie.service.DishService;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang.StringUtils;
import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.web.bind.annotation.*;

import java.util.List;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

/**
 * @Author 小杰 @Date 2022/4/28 17:55 @Version 1.0 菜品管理
 */
@RestController
@Slf4j
@RequestMapping("/dish")
public class DishController {
    @Autowired
    private DishService dishService;
    @Autowired
    private DishFlavorService dishFlavorService;
    @Autowired
    private CategoryService categoryService;
    @Autowired
    private RedisTemplate redisTemplate;

    /**
     * 新增菜品
     *
     * @param dishDto
     * @return
     */
    @PostMapping
    public R<String> save(@RequestBody DishDto dishDto) {
        log.info("新增菜品:{}", dishDto);
        dishService.saveWithFlavor(dishDto);
        //清除所有菜品的缓存数据。
//        Set keys = redisTemplate.keys("dish_*");
//        redisTemplate.delete(keys);
        //清除某个分类所对应的菜品数据。
        String key="dish_"+dishDto.getCategoryId()+"_1";
        redisTemplate.delete(key);
        return R.success("新增菜品成功");
    }

    /**
     * 菜品分页查询。
     *
     * @param page
     * @param pageSize
     * @param name
     * @return
     */
    @GetMapping("/page")
    public R<Page> page(int page, int pageSize, String name) {
        // 验证是否接收到前端数据。
        log.info("page:{},pagesize:{},name:{}", page, pageSize, name);
        // 构造分页构造器,
        Page<Dish> pageInfo = new Page<>(page, pageSize);
        Page<DishDto> dishDtoPage = new Page<>(page, pageSize);
        // 构造条件构造器,一定要添加泛型。
        LambdaQueryWrapper<Dish> queryWrapper = new LambdaQueryWrapper();
        // 添加过滤条件
        queryWrapper.like(StringUtils.isNotEmpty(name), Dish::getName, name);
        // 添加排序条件
        queryWrapper.orderByDesc(Dish::getUpdateTime);
        // 执行查询
        dishService.page(pageInfo, queryWrapper);
        // 对象的拷贝
        BeanUtils.copyProperties(pageInfo, dishDtoPage, "records");
        List<Dish> records = pageInfo.getRecords();
        // stream处理。
        List<DishDto> list =
                records.stream()
                        .map(
                                (dish) -> {
                                    DishDto dishDto = new DishDto();
                                    BeanUtils.copyProperties(dish, dishDto);
                                    // 分类id
                                    Long categoryId = dish.getCategoryId();
                                    // 根据id查分类对象。
                                    Category category = categoryService.getById(categoryId);
                                    // 获取categoryName
                                    if (category != null) {
                                        String categoryName = category.getName();
                                        dishDto.setCategoryName(categoryName);
                                    }
                                    return dishDto;
                                })
                        .collect(Collectors.toList());
        dishDtoPage.setRecords(list);
        return R.success(dishDtoPage);
    }

    /**
     * 根据id查询菜品信息。
     *
     * @param id
     * @return
     */
    @GetMapping("/{id}")
    public R<DishDto> get(@PathVariable Long id) {
        // 日志
        log.info("DishId:{}", id);
        // 执行查询。
        DishDto dishDto = dishService.getByIdWithFlavor(id);
        return R.success(dishDto);
    }

    /**
     * 更新菜品信息。
     *
     * @param dishDto
     * @return
     */
    @PutMapping
    public R<String> update(@RequestBody DishDto dishDto) {
        log.info("更新菜品:{}", dishDto);
        // 开始更新信息
        dishService.updateWithFlavor(dishDto);
        //清除所有菜品的缓存数据。
//        Set keys = redisTemplate.keys("dish_*");
//        redisTemplate.delete(keys);
        //清除某个分类所对应的菜品数据。
        String key="dish_"+dishDto.getCategoryId()+"_1";
        redisTemplate.delete(key);


        return R.success("更新菜品信息成功");
    }

    /**
     * 根据id删除一个或批量删除菜品。
     *
     * @param ids 待删除的菜品id。
     * @return
     */
    @DeleteMapping
    public R<String> delete(@RequestParam List<Long> ids) {
        // 增加日志验证是否接收到前端参数。
        log.info("根据id删除一个菜品:{}", ids);
        // 执行删除。
        dishService.deleteWithFlavor(ids);
        return R.success("删除菜品成功");
    }

    /**
     * 根据id修改菜品的状态status(停售和起售)
     *
     * <p>0停售,1起售。
     *
     * @param status
     * @param
     * @return
     */
    @PostMapping("/status/{status}")
    public R<String> updateStatusById(@PathVariable Integer status, @RequestParam List<Long> ids) {
        // 增加日志验证是否接收到前端参数。
        log.info("根据id修改菜品的状态:{},id为:{}", status, ids);
        // 通过id查询数据库。修改id为ids数组中的数据的菜品状态status为前端页面提交的status。
        UpdateWrapper<Dish> updateWrapper = new UpdateWrapper<>();
        updateWrapper.set("status", status).in("id", ids);
        // 更新状态。
        if (!dishService.update(updateWrapper)) {
            return R.error("修改菜品状态失败");
        }
        return R.success("修改菜品状态成功");
    }

    /* */
    /**
     * 根据条件查询对应的菜品信息。
     *
     * @param dish
     * @return
     */
  /*
  @GetMapping("/list")
  public R<List<Dish>> list(Dish dish) {
    log.info("接收参数:{}", dish);
    // 构造查询条件
    LambdaQueryWrapper<Dish> queryWrapper = new LambdaQueryWrapper<>();
    // 查询起售的。
    queryWrapper.eq(Dish::getStatus, 1);
    queryWrapper.eq(dish.getCategoryId() != null, Dish::getCategoryId, dish.getCategoryId());
    // 添加排序条件
    queryWrapper.orderByAsc(Dish::getSort).orderByDesc(Dish::getUpdateTime);
    // 执行查询
    List<Dish> dishList = dishService.list(queryWrapper);
    return R.success(dishList);
  }*/

    /**
     * 根据条件查询对应的菜品信息。
     *
     * @param dish
     * @return
     */
    @GetMapping("/list")
    public R<List<DishDto>> list(Dish dish) {
        List<DishDto> dishDtoList = null;
        String key = "dish_"+dish.getCategoryId() + "_" + dish.getStatus();
        //判断redis中是否有菜品数据,如果有则直接返回。
        dishDtoList = (List<DishDto>) redisTemplate.opsForValue().get(key);
        if (dishDtoList != null) {
            return R.success(dishDtoList);
        }

        log.info("接收参数:{}", dish);
        // 构造查询条件
        LambdaQueryWrapper<Dish> queryWrapper = new LambdaQueryWrapper<>();
        // 查询起售的。
        queryWrapper.eq(Dish::getStatus, 1);
        queryWrapper.eq(dish.getCategoryId() != null, Dish::getCategoryId, dish.getCategoryId());
        // 添加排序条件
        queryWrapper.orderByAsc(Dish::getSort).orderByDesc(Dish::getUpdateTime);
        // 执行查询
        List<Dish> dishList = dishService.list(queryWrapper);
        // 在原来的基础上追加flavor信息。
        // stream处理。
        dishDtoList =
                dishList.stream()
                        .map(
                                (item) -> {
                                    DishDto dishDto = new DishDto();
                                    BeanUtils.copyProperties(item, dishDto);
                                    // 分类id
                                    Long categoryId = item.getCategoryId();
                                    // 根据id查分类对象。
                                    Category category = categoryService.getById(categoryId);
                                    // 获取categoryName
                                    if (category != null) {
                                        String categoryName = category.getName();
                                        dishDto.setCategoryName(categoryName);
                                    }
                                    // 在原来的基础上追加flavor信息。
                                    Long dishId = item.getId();
                                    LambdaQueryWrapper<DishFlavor> lambdaqw = new LambdaQueryWrapper<>();
                                    lambdaqw.eq(DishFlavor::getDishId, dishId);
                                    List<DishFlavor> dishFlavors = dishFlavorService.list(lambdaqw);
                                    dishDto.setFlavors(dishFlavors);
                                    return dishDto;
                                })
                        .collect(Collectors.toList());
        //如果没有,就查询数据库,并把查询到的数据添加到缓存redis,并设置时间。
        redisTemplate.opsForValue().set(key, dishDtoList, 60, TimeUnit.MINUTES);
        return R.success(dishDtoList);
    }
}

测试通过:

  • 当前在v1.0分支,先把v1.0分支的内容commit并推送到远程库。
  • 使用git进行分支合并,需要先切换到master分支,然后把v1.0分支的优化部分合并到master。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值