• 基于Springboot外卖系统20:前端菜品展示+菜品数量查询


    1 菜品展示

    1.1 需求分析

    用户登录成功后跳转到系统首页,在首页需要根据分类来展示菜品和套餐。如果菜品设置了口味信息,需要展示 按钮,否则显示按钮。

     

    1.2 前端页面分析

    在开发代码之前,需要梳理一下前端页面和服务端的交互过程:

    1). 页面(front/index.html)发送ajax请求,获取分类数据(菜品分类和套餐分类)

    该功能已经实现了。通过请求响应的数据可以看到数据是可以正确获取到的。

    左侧的分类菜单,和右侧的菜品信息都可以看到,后续只需要将购物车列表的数据改成调用服务端接口查询即可。

    2). 页面发送ajax请求,获取第一个分类下的菜品或者套餐

     A. 根据分类ID查询套餐列表:

     B. 根据分类ID查询菜品列表:

     异步请求,查询分类对应的菜品列表已经实现了,但是查询的只是菜品的基本信息,不包含菜品的口味信息。所以在前端界面中看不到选择菜品分类的信息。

    经过上述的分析,服务端我们主要提供两个方法, 分别用来:

    A. 根据分类ID查询菜品列表(包含菜品口味列表), 具体请求信息如下:

    请求说明
    请求方式GET
    请求路径/dish/list
    请求参数?categoryId=1397844263642378242&status=1

    该功能在服务端已经实现,需要修改此方法,在原有方法的基础上增加查询菜品的口味信息。

    B. 根据分类ID查询套餐列表, 具体请求信息如下:

    请求说明
    请求方式GET
    请求路径/setmeal/list
    请求参数?categoryId=1397844263642378242&status=1

    该功能在服务端并未实现。

    2 代码开发

    2.1 查询菜品方法修改

    由于之前实现的根据分类查询菜品列表,仅仅查询了菜品的基本信息,未查询菜品口味信息,而移动端用户在点餐时,是需要选择口味信息的,所以需要对之前的代码实现进行完善。

    需要修改DishController的list方法,原来此方法的返回值类型为:R>。为了满足移动端对数据的要求(菜品基本信息和菜品对应的口味信息),现在需要将方法的返回值类型改为:R> ,因为在DishDto中封装了菜品对应的口味信息:

    代码逻辑:

    A. 根据分类ID查询,查询目前正在启售的菜品列表 (已实现)

    B. 遍历菜品列表,并查询菜品的分类信息及菜品的口味列表

    C. 组装数据DishDto,并返回

    1 DishController中增加根据条件查询对应的菜品数的list()方法

    1. package com.itheima.reggie.controller;
    2. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
    3. import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
    4. import com.itheima.reggie.common.R;
    5. import com.itheima.reggie.dto.DishDto;
    6. import com.itheima.reggie.entity.Category;
    7. import com.itheima.reggie.entity.Dish;
    8. import com.itheima.reggie.entity.DishFlavor;
    9. import com.itheima.reggie.service.CategoryService;
    10. import com.itheima.reggie.service.DishFlavorService;
    11. import com.itheima.reggie.service.DishService;
    12. import lombok.extern.slf4j.Slf4j;
    13. import org.springframework.beans.BeanUtils;
    14. import org.springframework.beans.factory.annotation.Autowired;
    15. import org.springframework.web.bind.annotation.*;
    16. import java.util.List;
    17. import java.util.stream.Collectors;
    18. /**
    19. * Description: 菜品管理 菜品及菜品口味的相关操作,统一使用这一个controller即可。
    20. * @version 1.0
    21. * @date 2022/8/18 11:08
    22. */
    23. @Slf4j
    24. @RestController
    25. @RequestMapping("/dish")
    26. public class DishController {
    27. @Autowired
    28. private DishService dishService;
    29. @Autowired
    30. private DishFlavorService dishFlavorService;
    31. @Autowired
    32. private CategoryService categoryService;
    33. @PostMapping
    34. public R save(@RequestBody DishDto dishDto){
    35. /**@Description: 新增菜品
    36. * @author LiBiGo
    37. * @date 2022/8/18 11:44
    38. */
    39. log.info(dishDto.toString());
    40. dishService.saveWithFlavor(dishDto);
    41. return R.success("新增菜品成功");
    42. }
    43. @GetMapping("/page")
    44. public R page(int page,int pageSize,String name){
    45. /**@Description: 菜品信息分页查询
    46. * @author LiBiGo
    47. *
    48. * 数据库查询菜品信息时,获取到的分页查询结果 Page 的泛型为 Dish,而最终需要给前端页面返回的类型为DishDto,
    49. * 所以这个时候就要进行转换,基本属性直接通过属性拷贝的形式对Page中的属性进行复制,
    50. * 对于结果列表 records属性需要进行特殊处理的(需要封装菜品分类名称);
    51. *
    52. * @date 2022/8/19 10:41
    53. */
    54. // 构造分页构造器对象
    55. Page pageInfo = new Page<>(page,pageSize);
    56. Page dishDtoPage = new Page<>();
    57. // 条件构造器
    58. LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>();
    59. // 添加过滤条件
    60. queryWrapper.like(name!=null,Dish::getName,name);
    61. // 添加排序条件
    62. queryWrapper.orderByDesc(Dish::getUpdateTime);
    63. // 执行分页查询
    64. dishService.page(pageInfo,queryWrapper);
    65. // 对象的拷贝
    66. BeanUtils.copyProperties(pageInfo,dishDtoPage,"records");
    67. List records = pageInfo.getRecords();
    68. List list = records.stream().map((item) -> {
    69. DishDto dishDto = new DishDto();
    70. BeanUtils.copyProperties(item,dishDto);
    71. Long categoryId = item.getCategoryId();//分类id
    72. //根据id查询分类对象
    73. Category category = categoryService.getById(categoryId);
    74. if(category != null){
    75. String categoryName = category.getName();
    76. dishDto.setCategoryName(categoryName);
    77. }
    78. return dishDto;
    79. }).collect(Collectors.toList());
    80. dishDtoPage.setRecords(list);
    81. return R.success(dishDtoPage);
    82. }
    83. @GetMapping("/{id}")
    84. public R get(@PathVariable Long id){
    85. /**@Description: 根据id查询菜品信息和对应的口味信息
    86. * @author LiBiGo
    87. * @date 2022/8/19 11:43
    88. */
    89. DishDto dishDto = dishService.getByIdWithFlavor(id);
    90. return R.success(dishDto);
    91. }
    92. @PutMapping
    93. // @PathVariable : 该注解可以用来提取url路径中传递的请求参数。
    94. public R update(@RequestBody DishDto dishDto){
    95. /**@Description: 修改菜品
    96. * @author LiBiGo
    97. * @date 2022/8/19 11:58
    98. */
    99. log.info(dishDto.toString());
    100. dishService.updateWithFlavor(dishDto);
    101. return R.success("新增菜品成功");
    102. }
    103. // @GetMapping("/list")
    104. // public R> list(Dish dish){
    105. // /**@Description: 根据条件查询对应的菜品数
    106. // * @author LiBiGo
    107. // * @date 2022/8/19 15:49
    108. // */
    109. // // 构造查询条件
    110. // LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>();
    111. // queryWrapper.eq(dish.getCategoryId()!=null,Dish::getCategoryId,dish.getCategoryId());
    112. // //添加条件,查询状态为1(起售状态)的菜品
    113. // queryWrapper.eq(Dish::getStatus,1);
    114. //
    115. // // 添加排序条件
    116. // queryWrapper.orderByDesc(Dish::getSort).orderByDesc(Dish::getUpdateTime);
    117. //
    118. // List list = dishService.list(queryWrapper);
    119. //
    120. // return R.success(list);
    121. // }
    122. @GetMapping("/list")
    123. public R> list(Dish dish){
    124. /**@Description: 根据条件查询对应的菜品数
    125. * @author LiBiGo
    126. * @date 2022/8/19 15:49
    127. */
    128. // 构造查询条件
    129. LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>();
    130. queryWrapper.eq(dish.getCategoryId()!=null,Dish::getCategoryId,dish.getCategoryId());
    131. //添加条件,查询状态为1(起售状态)的菜品
    132. queryWrapper.eq(Dish::getStatus,1);
    133. // 添加排序条件
    134. queryWrapper.orderByDesc(Dish::getSort).orderByDesc(Dish::getUpdateTime);
    135. List list = dishService.list(queryWrapper);
    136. List dishDtoList = list.stream().map((item) -> {
    137. DishDto dishDto = new DishDto();
    138. BeanUtils.copyProperties(item,dishDto);
    139. Long categoryId = item.getCategoryId();//分类id
    140. //根据id查询分类对象
    141. Category category = categoryService.getById(categoryId);
    142. if(category != null){
    143. String categoryName = category.getName();
    144. dishDto.setCategoryName(categoryName);
    145. }
    146. //当前菜品的id
    147. Long dishId = item.getId();
    148. LambdaQueryWrapper lambdaQueryWrapper = new LambdaQueryWrapper<>();
    149. lambdaQueryWrapper.eq(DishFlavor::getDishId,dishId);
    150. //SQL:select * from dish_flavor where dish_id = ?
    151. List dishFlavorList = dishFlavorService.list(lambdaQueryWrapper);
    152. dishDto.setFlavors(dishFlavorList);
    153. return dishDto;
    154. }).collect(Collectors.toList());
    155. return R.success(dishDtoList);
    156. }
    157. }

    2 在SetmealController中创建list()方法,根据条件查询套餐数据。

    1. package com.itheima.reggie.controller;
    2. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
    3. import com.baomidou.mybatisplus.extension.plugins.pagination.Page;
    4. import com.itheima.reggie.common.R;
    5. import com.itheima.reggie.dto.SetmealDto;
    6. import com.itheima.reggie.entity.Category;
    7. import com.itheima.reggie.entity.Setmeal;
    8. import com.itheima.reggie.service.CategoryService;
    9. import com.itheima.reggie.service.SetmealDishService;
    10. import com.itheima.reggie.service.SetmealService;
    11. import lombok.extern.slf4j.Slf4j;
    12. import org.apache.commons.lang.StringUtils;
    13. import org.springframework.beans.BeanUtils;
    14. import org.springframework.beans.factory.annotation.Autowired;
    15. import org.springframework.web.bind.annotation.*;
    16. import java.util.List;
    17. import java.util.stream.Collectors;
    18. /**
    19. * Description: 套餐管理
    20. * 不仅需要保存套餐的基本信息,还需要保存套餐关联的菜品数据,所以需要再该方法中调用业务层方法,完成两块数据的保存。
    21. * @version 1.0
    22. * @date 2022/8/19 15:37
    23. */
    24. @RestController
    25. @RequestMapping("/setmeal")
    26. @Slf4j
    27. public class SetmealController {
    28. @Autowired
    29. private SetmealService setmealService;
    30. @Autowired
    31. private CategoryService categoryService;
    32. @Autowired
    33. private SetmealDishService setmealDishService;
    34. @PostMapping
    35. // 页面传递的数据是json格式,需要在方法形参前面加上@RequestBody注解, 完成参数封装。
    36. public R save(@RequestBody SetmealDto setmealDto){
    37. /**@Description: 新增套餐
    38. * @version v1.0
    39. * @author LiBiGo
    40. * @date 2022/8/19 16:04
    41. */
    42. log.info("套餐信息:{}",setmealDto);
    43. setmealService.saveWithDish(setmealDto);
    44. return R.success("新增套餐成功");
    45. }
    46. @GetMapping("/page")
    47. public R page(int page,int pageSize,String name){
    48. /**@Description: 套餐分页查询
    49. * @author LiBiGo
    50. * @date 2022/8/21 10:40
    51. */
    52. // 分页构造器对象
    53. Page pageInfo = new Page<>(page,pageSize);
    54. Page dtoPage = new Page<>();
    55. LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>();
    56. // 添加查询条件,根据name进行like模糊查询
    57. queryWrapper.like(name!=null,Setmeal::getName,name);
    58. // 排序条件,根据更新时间进行降序排序
    59. queryWrapper.orderByDesc(Setmeal::getUpdateTime);
    60. setmealService.page(pageInfo,queryWrapper);
    61. // 拷贝对象
    62. BeanUtils.copyProperties(pageInfo,dtoPage,"record");
    63. List records = pageInfo.getRecords();
    64. List list = records.stream().map((item) -> {
    65. SetmealDto setmealDto = new SetmealDto();
    66. //对象拷贝
    67. BeanUtils.copyProperties(item,setmealDto);
    68. //分类id
    69. Long categoryId = item.getCategoryId();
    70. //根据分类id查询分类对象
    71. Category category = categoryService.getById(categoryId);
    72. if(category != null){
    73. //分类名称
    74. String categoryName = category.getName();
    75. setmealDto.setCategoryName(categoryName);
    76. }
    77. return setmealDto;
    78. }).collect(Collectors.toList());
    79. dtoPage.setRecords(list);
    80. return R.success(dtoPage);
    81. }
    82. @DeleteMapping
    83. public R delete(@RequestParam List ids){
    84. /**@Description: 删除套餐
    85. * @author LiBiGo
    86. * @date 2022/8/21 11:35
    87. */
    88. log.info("ids:{}",ids);
    89. setmealService.removeWithDish(ids);
    90. return R.success("套餐数据删除成功");
    91. }
    92. @GetMapping("/list")
    93. public R> list(Setmeal setmeal) {
    94. /**
    95. * 根据条件查询套餐数据
    96. * @param setmeal
    97. * @return
    98. */
    99. log.info("setmeal:{}", setmeal);
    100. //条件构造器
    101. LambdaQueryWrapper queryWrapper = new LambdaQueryWrapper<>();
    102. queryWrapper.like(StringUtils.isNotEmpty(setmeal.getName()), Setmeal::getName, setmeal.getName());
    103. queryWrapper.eq(null != setmeal.getCategoryId(), Setmeal::getCategoryId, setmeal.getCategoryId());
    104. queryWrapper.eq(null != setmeal.getStatus(), Setmeal::getStatus, setmeal.getStatus());
    105. queryWrapper.orderByDesc(Setmeal::getUpdateTime);
    106. return R.success(setmealService.list(queryWrapper));
    107. }
    108. }

    3 功能测试

    测试过程中可以使用浏览器的监控工具查看页面和服务端的数据交互细节。

     点击分类,根据分类查询菜品列表/套餐列表:

     

     

  • 相关阅读:
    WPF 简单的ComboBox自定义样式。
    pcl--第九节 点云分割
    Flink几个性能调优
    Android13---下拉状态栏添加阅读模式(MTK平台)
    linux阻塞IO与非阻塞IO在应用层和驱动层的实现(一)
    【NLP】LSTM 唐诗生成器 pytorch 版
    Python爬虫_案例分析(二)
    app查看 证书公钥和md5
    南京邮电大学统计学课程实验2 用EXCEL进行参数估计假设检验 指导
    python 正则表达式
  • 原文地址:https://blog.csdn.net/qq_39237205/article/details/126580288