Day7 苍穹外卖项目 缓存菜品、SpringCache框架、缓存套餐、添加购物车、查看购物车、清空购物车

目录

1.缓存菜品

1.1 问题说明

1.2 实现思路

1.3 代码开发

1.3.1 加入缓存

1.3.2 清除缓存

1.3.2.1 新增菜品优化

1.3.2.2 菜品批量删除优化

1.3.2.3 修改菜品优化

1.3.2.4 菜品起售停售优化

1.4 功能测试

1.4.1 加入缓存

1.4.2 菜品修改

1.5 代码提交

2.缓存套餐

2.1 Spring Cache

2.1.1 介绍

2.1.2 常用注解

2.1.3 入门案例

2.1.3.1 环境准备

2.1.3.2 @CachePut注解

2.1.3.3 @Cacheable注解

2.1.3.4 @CacheEvict注解

2.2 实现步骤

2.3 代码开发

2.3.1 导入maven坐标

2.3.2 启动类加@EnableCaching注解

2.3.3 加@Cacheable注解

2.3.4 加@CacheEvict注解

2.4 功能测试

2.5 代码提交

3.添加购物车

3.1 需求分析和设计

3.1.1 产品原型

3.1.2 接口设计

3.1.3 表设计

3.2 代码开发

3.2.1 DTO设计

3.2.2 Controller层

3.2.3 Service层接口

3.2.4 Service层实现类

3.2.5 Mapper层

3.3 功能测试

3.4 代码提交

4.查看购物车

4.1 需求分析和设计

4.1.1 产品原型

4.1.2 接口设计

4.2 代码开发

4.2.1 Controller层

4.2.2 Service层接口

4.2.3 Service层实现类

4.3 功能测试

4.4 代码提交

5.清空购物车

5.1 需求分析和设计

5.1.1 产品原型

5.1.2 接口设计

5.2 代码开发

5.2.1 Controller层

5.2.2 Service层接口

5.2.3 Service层实现类

5.2.4 Mapper层

5.3 功能测试


1.缓存菜品

1.1 问题说明

用户端小程序展示的菜品数据都是通过查询数据库获得,如果用户端访问量比较大,数据库访问压力随之增大。

结果:系统响应慢、用户体验差。

1.2 实现思路

通过Redis来缓存菜品数据,减少数据库查询操作。

缓存逻辑分析:

  • 每个分类下的菜品保存一份缓存数据

  • 数据库中菜品数据有变更时清理缓存数据

1.3 代码开发

1.3.1 加入缓存

修改用户端接口 DishController 的 list 方法,加入缓存处理逻辑:

	@Autowired
    private RedisTemplate redisTemplate;
	/**
     * 根据分类id查询菜品
     *
     * @param categoryId
     * @return
     */
    @GetMapping("/list")
    @ApiOperation("根据分类id查询菜品")
    public Result<List<DishVO>> list(Long categoryId) {

        //构造redis中的key,规则:dish_分类id
        String key = "dish_" + categoryId;

        //查询redis中是否存在菜品数据
        List<DishVO> list = (List<DishVO>) redisTemplate.opsForValue().get(key);
        if(list != null && list.size() > 0){
            //如果存在,直接返回,无须查询数据库
            return Result.success(list);
        }
		////////////////////////////////////////////////////////
        Dish dish = new Dish();
        dish.setCategoryId(categoryId);
        dish.setStatus(StatusConstant.ENABLE);//查询起售中的菜品

        //如果不存在,查询数据库,将查询到的数据放入redis中
        list = dishService.listWithFlavor(dish);
        ////////////////////////////////////////////////////////
        redisTemplate.opsForValue().set(key, list);

        return Result.success(list);
    }

1.3.2 清除缓存

为了保证数据库Redis中的数据保持一致,修改管理端接口 DishController 的相关方法,加入清理缓存逻辑。

需要改造的方法:

  • 新增菜品(删除dish_categoryId的缓存数据)

  • 修改菜品(删除有关dish_前缀的缓存数据)

  • 批量删除菜品(删除有关dish_前缀的缓存数据)

  • 起售、停售菜品(删除有关dish_前缀的缓存数据)

注意事项:如果加了缓存,要多查数据库,那就得不偿失。直接删除全部缓存、。

抽取清理缓存的方法:

在管理端DishController中添加

	@Autowired
    private RedisTemplate redisTemplate;
	/**
     * 清理缓存数据
     * @param pattern
     */
    private void cleanCache(String pattern){
        Set keys = redisTemplate.keys(pattern);
        redisTemplate.delete(keys);
    }

调用清理缓存的方法,保证数据一致性:

1.3.2.1 新增菜品优化

	/**
     * 新增菜品
     *
     * @param dishDTO
     * @return
     */
    @PostMapping
    @ApiOperation("新增菜品")
    public Result save(@RequestBody DishDTO dishDTO) {
        log.info("新增菜品:{}", dishDTO);
        dishService.saveWithFlavor(dishDTO);

        //清理缓存数据
        String key = "dish_" + dishDTO.getCategoryId();
        cleanCache(key);
        return Result.success();
    }

1.3.2.2 菜品批量删除优化

	/**
     * 菜品批量删除
     *
     * @param ids
     * @return
     */
    @DeleteMapping
    @ApiOperation("菜品批量删除")
    public Result delete(@RequestParam List<Long> ids) {
        log.info("菜品批量删除:{}", ids);
        dishService.deleteBatch(ids);

        //将所有的菜品缓存数据清理掉,所有以dish_开头的key
        cleanCache("dish_*");

        return Result.success();
    }

1.3.2.3 修改菜品优化

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

        //将所有的菜品缓存数据清理掉,所有以dish_开头的key
        cleanCache("dish_*");

        return Result.success();
    }

1.3.2.4 菜品起售停售优化

	/**
     * 菜品起售停售
     *
     * @param status
     * @param id
     * @return
     */
    @PostMapping("/status/{status}")
    @ApiOperation("菜品起售停售")
    public Result<String> startOrStop(@PathVariable Integer status, Long id) {
        dishService.startOrStop(status, id);

        //将所有的菜品缓存数据清理掉,所有以dish_开头的key
        cleanCache("dish_*");

        return Result.success();
    }

1.4 功能测试

可以通过如下方式进行测试:

  • 查看控制台sql

  • 前后端联调

  • 查看Redis中的缓存数据

加入缓存菜品修改两个功能测试为例,通过前后端联调方式,查看控制台sql的打印和Redis中的缓存数据变化。

1.4.1 加入缓存

当第一次查询某个分类的菜品时,会从数据为中进行查询,同时将查询的结果存储到Redis中,在后绪的访问,若查询相同分类的菜品时,直接从Redis缓存中查询,不再查询数据库。

登录小程序:选择蜀味牛蛙(id=17)

查看控制台sql:有查询语句,说明是从数据库中进行查询。

查看Redis中的缓存数据:说明缓存成功。

再次访问:选择蜀味牛蛙(id=17)

说明是从Redis中查询的数据。

1.4.2 菜品修改

当在后台修改菜品数据时,为了保证Redis缓存中的数据和数据库中的数据时刻保持一致,当修改后,需要清空对应的缓存数据。用户再次访问时,还是先从数据库中查询,同时再把查询的结果存储到Redis中,这样,就能保证缓存和数据库的数据保持一致。

进入后台:修改蜀味牛蛙分类下的任意一个菜品,当前分类的菜品数据已在Redis中缓存

修改:

查看Redis中的缓存数据:说明修改时,已清空缓存。

用户再次访问同一个菜品分类时,需要先查询数据库,再把结果同步到Redis中,保证了两者数据一致性。

其它功能测试步骤基本一致,自已测试即可。

1.5 代码提交

2.缓存套餐

2.1 Spring Cache

2.1.1 介绍

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

在spring boot项目中,使用缓存技术只需在项目中导入相关缓存技术的依赖包,并在启动类上使用@EnableCaching开启缓存支持即可。

Spring Cache 提供了一层抽象,底层可以切换不同的缓存实现,例如:

  • EHCache

  • Caffeine

  • Redis(常用)

注意事项:切换缓存实现,只需要引入其maven坐标,就能实现。

起步依赖:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-cache</artifactId>                                                           
    <version>2.7.3</version> 
</dependency>
### 关于黑马苍穹外卖 Project Day04 菜品起售停售功能 对于黑马苍穹外卖项目的第4天课程,其中涉及到了菜品的起售与停售功能的设计与实现。此部分通常涉及到数据库设计以及前后端交互逻辑。 #### 数据库设计 为了支持菜品状态的变化(即是否可销售),一般会在菜品表中增加一个字段用于表示该商品的状态。例如,在MySQL中创建菜品表时可能会这样定义: ```sql CREATE TABLE dishes ( id INT AUTO_INCREMENT PRIMARY KEY, name VARCHAR(255) NOT NULL, price DECIMAL(10, 2), status TINYINT DEFAULT 1 COMMENT '1=on sale; 0=out of stock' ); ``` 这里`status`列用来标记菜品当前是否处于售卖状态[^1]。 #### 前端展示控制 前端页面应当能够动态反映后台数据变化情况。当管理员更改某道菜的状态后,前台顾客看到的信息也应随之更新。可以通过AJAX请求定期向服务器查询最新菜单列表并刷新视图来达成这一目标。 #### 后台管理接口开发 针对菜品上下架操作,需要提供相应的API供管理人员调用。这些API应该接受参数如dishId和newStatus,并据此修改对应的记录项。下面是一个简单的Spring Boot风格控制器方法示例: ```java @RestController @RequestMapping("/admin/dishes") public class DishAdminController { @Autowired private DishService dishService; /** * 更新指定ID菜品的状态. */ @PutMapping("/{id}/toggle-status") public ResponseEntity<String> toggleDishStatus(@PathVariable Long id, @RequestParam Integer newStatus){ try { boolean success = this.dishService.updateDishStatus(id,newStatus); if (success){ return ResponseEntity.ok("Updated successfully"); }else{ throw new RuntimeException(); } } catch (Exception e){ log.error(e.getMessage(),e); return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).body("Failed to update."); } } } ``` 上述代码片段展示了如何构建RESTful API以允许通过PUT请求改变特定菜品的状态。 #### 客户端交互逻辑优化建议 考虑到用户体验,可以在用户点击“下架”按钮前弹窗提示确认动作;另外还可以加入防抖动机制防止短时间内频繁提交相同命令造成不必要的网络流量浪费。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值