2.1 需求分析
后台系统中可以管理菜品信息,通过新增功能来添加一个新的菜品,在添加菜品时需要选择当前菜品所属的菜品分类,并且需要上传菜品图片,在移动端会按照菜品分类来展示对应的菜品信息 。
2.2 数据模型
新增菜品,其实就是将新增页面录入的菜品信息插入到dish表,如果添加了口味做法,还需要向dish_flavor表插入数据。
表结构 | 说明 |
dish | 菜品表 |
dish_flavor | 菜品口味表 |
1). 菜品表:dish
2). 菜品口味表:dish_flavor
2.3 准备工作
1). 实体类 DishFlavor
package com.itheima.reggie.entity; import com.baomidou.mybatisplus.annotation.FieldFill; import com.baomidou.mybatisplus.annotation.IdType; import com.baomidou.mybatisplus.annotation.TableField; import com.baomidou.mybatisplus.annotation.TableId; import lombok.Data; import java.io.Serializable; import java.time.LocalDateTime; /** 菜品口味 */ @Data public class DishFlavor implements Serializable { private static final long serialVersionUID = 1L; private Long id; //菜品id private Long dishId; //口味名称 private String name; //口味数据list private String value; @TableField(fill = FieldFill.INSERT) private LocalDateTime createTime; @TableField(fill = FieldFill.INSERT_UPDATE) private LocalDateTime updateTime; @TableField(fill = FieldFill.INSERT) private Long createUser; @TableField(fill = FieldFill.INSERT_UPDATE) private Long updateUser; //是否删除 private Integer isDeleted; }
2). Mapper接口DishFlavorMapper
package com.itheima.reggie.mapper; import com.baomidou.mybatisplus.core.mapper.BaseMapper; import com.itheima.reggie.entity.DishFlavor; import org.apache.ibatis.annotations.Mapper; @Mapper public interface DishFlavorMapper extends BaseMapper<DishFlavor> { }
3). 业务层接口 DishFlavorService
package com.itheima.reggie.service; import com.baomidou.mybatisplus.extension.service.IService; import com.itheima.reggie.entity.DishFlavor; public interface DishFlavorService extends IService<DishFlavor> { }
4). 业务层实现类 DishFlavorServiceImpl
package com.itheima.reggie.service.impl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.itheima.reggie.entity.DishFlavor; import com.itheima.reggie.mapper.DishFlavorMapper; import com.itheima.reggie.service.DishFlavorService; import org.springframework.stereotype.Service; /** * Description: new java files header.. * * @author w * @version 1.0 * @date 2022/8/18 11:07 */ @Service public class DishFlavorServiceImpl extends ServiceImpl<DishFlavorMapper, DishFlavor> implements DishFlavorService { }
2.4 前端页面分析
2.4.1 新增菜品时前端页面和服务端的交互过程
1). 点击新建菜品按钮, 访问页面(backend/page/food/add.html), 页面加载时发送ajax请求,请求服务端获取菜品分类数据并展示到下拉框中
2). 页面发送请求进行图片上传,请求服务端将图片保存到服务器(上传功能已实现)
3). 页面发送请求进行图片下载,将上传的图片进行回显(下载功能已实现)
4). 点击保存按钮,发送ajax请求,将菜品相关数据以json形式提交到服务端
页面代码:
浏览器抓取请求:
开发新增菜品功能,其实就是在服务端编写代码去处理前端页面发送的这4次请求(上传、下载已实现)即可。经过上述的分析,还需要在服务端实现两块功能:
A. 菜品分类数据列表查询, 具体请求信息整理如下 :
请求 | 说明 |
请求方式 | GET |
请求路径 | /category/list |
请求参数 | ?type=1 |
B. 保存菜品信息, 具体请求信息整理如下 :
请求 | 说明 |
请求方式 | POST |
请求路径 | /dish |
请求参数 | json格式 |
2.5 代码实现
2.5.1 菜品分类查询
在CategoryController中增加方法实现菜品分类查询,根据分类进行查询,并对查询的结果按照sort排序字段进行升序排序,如果sort相同,再按照修改时间倒序排序。
package com.itheima.reggie.controller; import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper; import com.baomidou.mybatisplus.extension.plugins.pagination.Page; import com.itheima.reggie.common.R; import com.itheima.reggie.entity.Category; import com.itheima.reggie.service.CategoryService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; import java.util.List; /** * Description: 分类管理 */ @RestController @RequestMapping("/category") @Slf4j public class CategoryController { @Autowired private CategoryService categoryService; /**@Description: 新增分类 * @author LiBiGo * @date 2022/8/15 14:05 */ @PostMapping public R<String> save(@RequestBody Category category){ log.info("category:{}",category); categoryService.save(category); return R.success("新增分类成功"); } @GetMapping("/page") public R<Page> page(int page,int pageSize){ /**@Description: 分页查询 * @author LiBiGo * @date 2022/8/15 14:21 */ // 分页构造 Page<Category> pageinfo = new Page<>(page,pageSize); // 构造条件构造器对象 LambdaQueryWrapper<Category> queryWrapper = new LambdaQueryWrapper(); // 添加排序条件,根据sore进行排序 queryWrapper.orderByAsc(Category::getSort); // 进行分页查询 categoryService.page(pageinfo,queryWrapper); return R.success(pageinfo); } @DeleteMapping public R<String> delete(Long id){ /**@Description: 根据id删除分类 * @author LiBiGo * @date 2022/8/16 9:58 */ log.info("删除分类,id为{}",id); categoryService.remove(id); return R.success("分类信息删除成功"); } @PutMapping public R<String> update(@RequestBody Category category){ /**@Description: 根据id修改分类信息 * @author LiBiGo * @date 2022/8/16 10:49 */ log.info("根据id修改分类信息:{}",category); categoryService.updateById(category); return R.success("修改分类成功"); } @GetMapping("/list") public R<List<Category>> list(Category category){ /**@Description: 根据条件查询分类数据 * @author LiBiGo * @date 2022/8/18 11:24 */ // 条件构造器 LambdaQueryWrapper<Category> queryWrapper = new LambdaQueryWrapper<>(); // 添加条件 queryWrapper.eq(category.getType() != null,Category::getType,category.getType()); // 添加排序条件 queryWrapper.orderByAsc(Category::getSort).orderByDesc(Category::getUpdateTime); List<Category> list = categoryService.list(queryWrapper); return R.success(list); } }
代码编写完毕之后,我们可以打开新增菜品页面,查看响应的数据,及页面下拉列表的渲染情况:
2.5.2 保存菜品信息
在上述的分析中,在保存菜品页面传递过来的是json格式数据,格式如下:
{ "name":"佛跳墙", "price":88800, "code":"", "image":"da9e1c70-fc32-4781-9510-a1c4ccd2ff59.jpg", "description":"佛跳墙", "status":1, "categoryId":"1397844357980663809", "flavors":[ { "name":"辣度", "value":"[\"不辣\",\"微辣\",\"中辣\",\"重辣\"]", "showOption":false }, { "name":"忌口", "value":"[\"不要葱\",\"不要蒜\",\"不要香菜\",\"不要辣\"]", "showOption":false } ] }
存在问题:如果使用菜品类Dish来封装,只能封装菜品的基本属性,flavors属性是无法封装的。
解决方案:需要自定义一个实体类,然后继承自 Dish,并对Dish的属性进行拓展,增加 flavors 集合属性(内部封装DishFlavor)。
1). 导入 DishDto 实体类
封装页面传递的请求参数。
package com.itheima.reggie.dto; import com.itheima.reggie.entity.Dish; import com.itheima.reggie.entity.DishFlavor; import lombok.Data; import java.util.ArrayList; import java.util.List; @Data public class DishDto extends Dish { private List<DishFlavor> flavors = new ArrayList<>(); private String categoryName; private Integer copies; }
拓展: 我们在做项目时,经常会涉及到各种类型的实体模型。基本包含以下几种
实体模型 | 描述 |
DTO | Data Transfer Object(数据传输对象),一般用于展示层与服务层之间的数据传输。 |
Entity | 最常用实体类,基本和数据表一一对应,一个实体类对应一张表。 |
VO | Value Object(值对象), 主要用于封装前端页面展示的数据对象,用一个VO对象来封装整个页面展示所需要的对象数据 |
PO | Persistant Object(持久层对象), 是ORM(Objevt Relational Mapping)框架中Entity,PO属性和数据库中表的字段形成一一对应关系 |
2). DishController定义方法新增菜品
在该Controller的方法中,不仅需要保存菜品的基本信息,还需要保存菜品的口味信息,需要操作两张表,所以需要在DishService接口中定义接口方法,在这个方法中需要保存上述的两部分数据。
package com.itheima.reggie.controller; import com.fasterxml.jackson.databind.annotation.JsonSerialize; import com.itheima.reggie.common.R; import com.itheima.reggie.dto.DishDto; import com.itheima.reggie.service.DishFlavorService; import com.itheima.reggie.service.DishService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestBody; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RestController; /** * Description: 菜品管理 菜品及菜品口味的相关操作,统一使用这一个controller即可。 * @version 1.0 * @date 2022/8/18 11:08 */ @Slf4j @RestController @RequestMapping("/dish") public class DishController { @Autowired private DishService dishService; @Autowired private DishFlavorService dishFlavorService; @PostMapping public R<String> save(@RequestBody DishDto dishDto){ /**@Description: 新增菜品 * @author LiBiGo * @date 2022/8/18 11:44 */ log.info(dishDto.toString()); dishService.saveWithFlavor(dishDto); return R.success("新增菜品成功"); } }
3). DishService中增加方法saveWithFlavor
package com.itheima.reggie.service; import com.baomidou.mybatisplus.extension.service.IService; import com.itheima.reggie.dto.DishDto; import com.itheima.reggie.entity.Dish; public interface DishService extends IService<Dish> { // 新增菜品,需要同时插入菜品对应的口味数据,需要操作两张表,dish、dish_flavor public void saveWithFlavor(DishDto dishDto); }
4). DishServiceImpl中实现方法saveWithFlavor
页面传递的菜品口味信息,仅仅包含name 和 value属性,缺少一个非常重要的属性dishId, 所以在保存完菜品的基本信息后,我们需要获取到菜品ID,然后为菜品口味对象属性dishId赋值。
具体逻辑如下:
①. 保存菜品基本信息 ;
②. 获取保存的菜品ID ;
③. 获取菜品口味列表,遍历列表,为菜品口味对象属性dishId赋值;
④. 批量保存菜品口味列表;
代码实现如下:
package com.itheima.reggie.service.impl; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.itheima.reggie.dto.DishDto; import com.itheima.reggie.entity.Dish; import com.itheima.reggie.entity.DishFlavor; import com.itheima.reggie.mapper.DishMapper; import com.itheima.reggie.service.DishFlavorService; import com.itheima.reggie.service.DishService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import java.util.List; import java.util.stream.Collectors; /** * Description: new java files header.. * * @author w * @version 1.0 * @date 2022/8/16 10:15 */ @Service @Slf4j public class DishServiceImpl extends ServiceImpl<DishMapper,Dish> implements DishService{ @Autowired private DishFlavorService dishFlavorService; @Override @Transactional public void saveWithFlavor(DishDto dishDto) { /**@Description: 新增菜品 同时保存对应的口味数据 * @author LiBiGo * @date 2022/8/18 11:58 */ // 保存菜品的基本信息到菜品表dish this.save(dishDto); Long dishId = dishDto.getId(); // 菜品id // 菜品口味 List<DishFlavor> flavors = dishDto.getFlavors(); flavors = flavors.stream().map((item)->{ item.setDishId(dishId); return item; }).collect(Collectors.toList()); // 保存菜品口味数据到菜品口味表dish_flavor dishFlavorService.saveBatch(flavors); } }
5). 在引导类上加注解 @EnableTransactionManagement
Service层方法上加的注解@Transactional要想生效,需要在引导类上加上注解 @EnableTransactionManagement, 开启对事务的支持。
package com.itheima.reggie; import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.web.servlet.ServletComponentScan; import org.springframework.transaction.annotation.EnableTransactionManagement; @Slf4j //是lombok中提供的注解, 用来通过slf4j记录日志。 @SpringBootApplication // 启动类 // 需要在引导类上,加上Servlet组件扫描的注解,来扫描过滤器配置的@WebFilter注解,扫描后,使过滤器在运行时生效。 @ServletComponentScan // 扫描拦截器注解 @EnableTransactionManagement public class ReggieApplication { /**@Description: 当搭建完上述的基础环境之后, 就可以通过引导类, 启动该项目。 * @author LiBiGo * @date 2022/8/12 0:23 */ public static void main(String[] args) { SpringApplication.run(ReggieApplication.class,args); log.info("项目启动成功............"); } }
2.6 功能测试
代码编写完毕之后,我们重新启动服务,访问项目,然后登陆到系统中,进行菜品的新增测试,在测试时,我们可以通过debug断点跟踪的形式,查看我们传输的数据,及数据的封装。
debug跟踪数据的封装情况:
然后在测试完毕后, 我们可以检查一下数据库中的数据保存情况: