java项目实战--瑞吉外卖Day4

java项目实战–瑞吉外卖Day4


一、文件上传下载

1.文件上传介绍

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2.文件下载介绍

在这里插入图片描述

3.文件上传代码实现

在进行文件上传和下载测试时需要先登录或者在全局异常处理类LoginCheckFilter中添加/common/**使得上传和下载不会被过滤器拦截:
在这里插入图片描述

在这里插入图片描述
在controller包中创建CommonController类:

在这里插入图片描述
在CommonController类中创建upload方法实现文件上传功能,代码如下:

package com.mj.reggie.controller;

import com.mj.reggie.common.R;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;

import java.io.File;
import java.io.IOException;
import java.util.UUID;

/**
 * @author mj
 * @version 1.0
 * 文件上传和下载
 */
@RestController
@RequestMapping("/common")
@Slf4j
public class CommonController {

    @Value("${reggie.path}")
    private String basePath;
    /*
     * 文件上传
     * */
    @PostMapping("/upload")
    public R<String> upload(MultipartFile file) {
//        file是一个临时文件,需要转存到指定位置,否则在请求完成后临时文件会被删除
        log.info(file.toString());
//        获取上传文件时的原始文件名,但是不建议使用这种方式,因为有可能重名,后面的文件会覆盖前面的
        String originalFilename = file.getOriginalFilename();
//        截取原始文件名的后缀
        String suffix = originalFilename.substring(originalFilename.lastIndexOf("."));
//        使用UUID随机生成字符串,防止重复命名问题
        String fileName = UUID.randomUUID().toString() + suffix;
//        创建一个目录对象
        File dir = new File(basePath);
//        判断当前目录是否存在
        if (!dir.exists()){
//            目录不存在,需要创建
            dir.mkdirs();
        }

//        将临时文件转存到指定位置
        try {
            file.transferTo(new File(basePath + fileName));
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
        return R.success(fileName);
    }
}

4.文件下载代码实现

在这里插入图片描述
在CommonController类中创建download方法实现文件下载功能,代码如下:

    /*
     * 文件下载
     * */
    @GetMapping("/download")
    public void download(String name, HttpServletResponse response) {
        try {

//                    输入流,通过输入流读取文件内容
            FileInputStream fileInputStream = new FileInputStream(new File(basePath + name));
//                    输出流,通过输出流将文件写回浏览器,在浏览器展示图片
            ServletOutputStream outputStream = response.getOutputStream();

            response.setContentType("/image/jpeg");

            int len = 0;
            byte[] bytes = new byte[1024];
            while ((len = fileInputStream.read(bytes)) != -1){
                outputStream.write(bytes,0,len);
                outputStream.flush();
            }
//            关闭资源
            outputStream.close();
            fileInputStream.close();

        } catch (Exception e) {

            e.printStackTrace();
        }

    }

二、新增菜品

1.需求分析

在这里插入图片描述

2.数据模型

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

2.代码开发

准备工作:
在这里插入图片描述

先将资料中的实体类DoshFlavor粘贴到entity包中:
在这里插入图片描述
在mapper包下创建DishFlavorMapper接口:
在这里插入图片描述
在service包下创建DishFlavorService接口:
在这里插入图片描述
在service.impl包下创建DishFlavorServiceImpl实现类:
在这里插入图片描述
在controller包下创建DishController类:
在这里插入图片描述

梳理交互过程:
在这里插入图片描述
1、先在CategoryController类中创建list方法,实现当页面(backend/page/food/add.html)发送ajax请求,请求服务端获取菜品分类数据并展示到下拉框中这一功能:

在这里插入图片描述
代码如下:

    /*
     * 根据条件查询分类数据
     * */
    @GetMapping("list")
    public R<List<Category>> list(Category category) {
//        条件构造器
        LambdaQueryWrapper<Category> queryWrapper = new LambdaQueryWrapper<>();
//        添加条件
        queryWrapper.eq(category.getType() != null, Category::getType, category.getType());
//        添加排序条件
        queryWrapper.orderByAsc(Category::getSort).orderByDesc(Category::getCreateTime);

        List<Category> list = categoryService.list(queryWrapper);
        return R.success(list);
    }

在实现添加菜品的保存功能时,因为页面提交的数据中有flavors,而dish实体类中并没有flavors这一属性
在这里插入图片描述
所以我们需要导入DTO
在这里插入图片描述
创建一个dto包,将资料中的DishDto类粘贴到dto包中:

在这里插入图片描述

在这里插入图片描述

导入完DishDto类后,在DishService接口中声明saveWithFlavor方法用于在dish和dish_flavor表中同时插入数据:
在这里插入图片描述
在DishServiceImpl类中实现saveWithFlavor方法:
在这里插入图片描述
代码如下:

package com.mj.reggie.service.impl;

import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
import com.mj.reggie.dto.DishDto;
import com.mj.reggie.entity.Dish;
import com.mj.reggie.entity.DishFlavor;
import com.mj.reggie.mapper.DishMapper;
import com.mj.reggie.service.DishFlavorService;
import com.mj.reggie.service.DishService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.List;
import java.util.stream.Collectors;

/**
 * @author mj
 * @version 1.0
 */
@Slf4j
@Service
@Transactional  //事务控制
public class DishServiceImpl extends ServiceImpl<DishMapper, Dish> implements DishService {

	@Autowired
    private DishFlavorService dishFlavorService;

    /*
    新增菜品,同时保存对应的口味数据
    * */
    @Override
    public void saveWithFlavor(DishDto dishDto) {
//      保存菜品的基本信息到菜品表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);
    }
}

同时在ReggieApplication类中添加EnableTransactionManagement注解来开启事务支持:
在这里插入图片描述
最后在DishController类中创建save方法实现新增菜品功能
在这里插入图片描述
代码如下:

package com.mj.reggie.controller;

import com.mj.reggie.common.R;
import com.mj.reggie.dto.DishDto;
import com.mj.reggie.service.DishFlavorService;
import com.mj.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;

/**
 * @author mj
 * @version 1.0
 * 菜品管理
 */
@RestController
@RequestMapping("/dish")
@Slf4j
public class DishController {
    @Autowired
    private DishService dishService;
    @Autowired
    private DishFlavorService dishFlavorService;

    /*
     * 新增菜品
     * */
    @PostMapping
    public R<String> save(@RequestBody DishDto dishDto) {
        log.info(dishDto.toString());
        dishService.saveWithFlavor(dishDto);
        return R.success("新增菜品成功");
    }
}

三、菜品信息分页查询

1.需求分析

在这里插入图片描述

2.代码开发

梳理交互过程:
在这里插入图片描述

在实现菜品信息分页查询功能之前,我们先在DishController类中创建一个categoryService对象:
在这里插入图片描述
接下来我们在DishController类中实现菜品信息分页查询功能,代码如下:

    /*
     * 菜品信息分页查询
     * */
    @GetMapping("/page")
    public R<Page> page(int page, int pageSize, String name) {
//      构造分页构造器对象
        Page<Dish> pageInfo = new Page<>(page, pageSize);
        Page<DishDto> dishDtoPage = new Page<>();

//      条件构造器
        LambdaQueryWrapper<Dish> lambdaQueryWrapper = new LambdaQueryWrapper<>();

//      添加构造条件
        lambdaQueryWrapper.like(name != null, Dish::getName, name);

//        添加排序条件,按照更新时间进行降序排序
        lambdaQueryWrapper.orderByDesc(Dish::getUpdateTime);

//        执行分页查询
        dishService.page(pageInfo, lambdaQueryWrapper);

//        对象拷贝
        BeanUtils.copyProperties(pageInfo,dishDtoPage,"records");

        List<Dish> records = pageInfo.getRecords();
        List<DishDto> list = records.stream().map((item)->{
            DishDto dishDto = new DishDto();

            BeanUtils.copyProperties(item,dishDto);

            Long categoryId = item.getCategoryId(); //分类id
//            根据id查询分类对象
            Category category = categoryService.getById(categoryId);

            if (category != null) {
                String categoryName = category.getName();
                dishDto.setCategoryName(categoryName);
            }

            return dishDto;
        }).collect(Collectors.toList());

        dishDtoPage.setRecords(list);

        return R.success(dishDtoPage);
    }

四、修改菜品

1.需求分析在这里插入图片描述

2.代码开发

梳理交互过程:
在这里插入图片描述
因为1,3功能我们之前已经实现了,所以现在只需要再实现2,4功能。我们先实现功能2,根据id查询当前菜品信息,用于菜品信息回显这一功能:
在这里插入图片描述

在DishService接口类中创建getByIdWithFlavor方法:
在这里插入图片描述
然后在DishServiceImpl类中实现getByIdWithFlavor方法,代码如下:

    /*
     * 根据id查询对应的菜品信息和口味信息
     * */
    public DishDto getByIdWithFlavor(Long id) {
//        查询菜品的基本信息,通过dish表查询
        Dish dish = this.getById(id);
//        new一个dishDto对象用于拷贝
        DishDto dishDto = new DishDto();
//        将dish中的基本信息拷贝到dishDto对象中
        BeanUtils.copyProperties(dish,dishDto);

//        查询当前菜品对应的口味信息,从dish_flavor表查询
        LambdaQueryWrapper<DishFlavor> lambdaQueryWrapper = new LambdaQueryWrapper<>();
        lambdaQueryWrapper.eq(DishFlavor::getDishId,dish.getId());
        List<DishFlavor> flavors = dishFlavorService.list(lambdaQueryWrapper);
        dishDto.setFlavors(flavors);
        return dishDto;
    }

最后在DishController类中调用getByIdWithFlavor方法实现根据id查询菜品信息和对应的口味信息功能,代码如下:

    /*
     * 根据id查询菜品信息和对应的口味信息
     * */
    @GetMapping("/{id}")  //当请求路径中有id这种参数时,需要使用PathVariable注解
    public R<DishDto> get(@PathVariable Long id) {

        DishDto dishDto = dishService.getByIdWithFlavor(id);

        return R.success(dishDto);
    }

接下来实现功能4,修改菜品后的保存功能:

在这里插入图片描述
在DishService接口中创建updateWithFlavor方法:
在这里插入图片描述
在DishServiceImpl类中实现updateWithFlavor方法:
在这里插入图片描述
代码如下:

    @Override
    @Transactional //事务注解
    public void updateWithFlavor(DishDto dishDto) {
//        更新dish表基本信息
        this.updateById(dishDto);
//        清理当前菜品对应的口味数据--dish_flavor表的delete操作
        LambdaQueryWrapper<DishFlavor> queryWrapper = new LambdaQueryWrapper();
        queryWrapper.eq(DishFlavor::getDishId,dishDto.getId());

        dishFlavorService.remove(queryWrapper);
//        添加当前提交过来的口味数据--dish_flavor表的insert操作
        List<DishFlavor> flavors = dishDto.getFlavors();

        flavors = flavors.stream().map((item) -> {
            item.setDishId(dishDto.getId());
            return item;
        }).collect(Collectors.toList());

        dishFlavorService.saveBatch(flavors);
    }

最后在DishController类中调用updateWithFlavor方法实现修改菜品功能,代码如下:

    /*
     * 修改菜品
     * */
    @PutMapping
    public R<String> update(@RequestBody DishDto dishDto) {
        log.info(dishDto.toString());
        dishService.updateWithFlavor(dishDto);
        return R.success("新增菜品成功");
    }
### 回答1: 瑞吉外卖是一个基于Java开发的项目实战,适用于在线外卖订餐系统。该项目提供了用户注册、登录、浏览餐厅、查看菜单、下订单等功能。 首先,我们需要在网盘上下载瑞吉外卖的项目源代码文件。通过提供的下载链接,我们可以将项目源代码文件下载到本地。下载完成后,我们可以将文件解压缩,并使用Java开发工具(如Eclipse或IntelliJ IDEA)导入项目。 接下来,我们需要安装项目所需的Java开发环境。确保已经安装了JDK(Java Development Kit)和Maven(项目构建工具)。这样可以保证项目能够正常编译和运行。 在导入项目后,我们可以查看项目的目录结构。主要包括源代码、配置文件和静态资源文件等。在源代码文件夹中,我们可以找到各种Java类文件,包括控制器、实体类、服务类等。配置文件夹中包含项目的配置文件,用于配置数据库连接、日志记录等。静态资源文件夹中包含了项目所需的各种图片、样式表和JavaScript文件等。 在开始开发之前,我们需要先配置数据库。将提供的SQL脚本文件导入到MySQL数据库中,并在项目配置文件中修改数据库连接相关的配置信息。 接下来,我们可以根据需求对项目进行开发和定制化。例如,我们可以根据需要添加更多的功能模块,如优惠券管理、配送员管理等。我们也可以根据需求修改前端页面的样式和布局,以满足用户的需求。 开发完成后,我们可以使用Maven将项目打包成可执行的WAR文件。将WAR文件上传至服务器,并部署在Tomcat等Java Web服务器上。通过访问服务器的IP地址和端口号,我们就可以在浏览器中访问瑞吉外卖系统了。 总之,下载并实战瑞吉外卖项目需要下载源代码文件,并在Java开发工具中导入项目。然后,我们可以根据需求进行开发和定制化,并最终将项目打包部署在服务器上。最后,我们可以通过浏览器访问项目,体验瑞吉外卖系统的功能。 ### 回答2: 瑞吉外卖是一个基于Java语言开发的项目实战,项目的主要目标是实现一个在线外卖订餐系统。用户可以通过网页或手机应用程序浏览餐厅菜单、下订单、查看订单状态等功能。 该项目的开发环境主要包括Java SE、Java EE、Spring框架和MySQL数据库。其中,Java SE用于实现基本的语言特性和数据处理操作,Java EE用于构建Web应用程序,Spring框架用于实现系统的MVC架构,MySQL数据库用于存储用户信息、菜品信息和订单数据等。 项目的实施步骤如下: 1. 需求分析:首先,根据用户的需求分析,确定项目的基本功能和需求。 2. 系统设计:基于需求分析的结果,进行系统设计,包括数据库设计、界面设计和系统架构设计等。 3. 环境搭建:安装配置Java开发环境,包括JDK、开发工具(如Eclipse或IntelliJ IDEA)、Web服务器(如Tomcat)和数据库管理系统(MySQL)。 4. 数据库建模:创建数据库表结构,定义各个表之间的关系。 5. 编码实现:根据系统设计的结果,进行编码实现,包括前端界面的开发和后端功能的开发。 6. 软件测试:对已实现的功能进行测试,包括单元测试、集成测试和系统测试等,保证系统的稳定性和可靠性。 7. 部署上线:将项目部署到服务器上,使用户可以通过网络访问系统。 8. 运维和优化:监控系统运行情况,对性能进行优化和改进。 最后,用户可以通过网盘下载瑞吉外卖的源代码和相关文档,以便学习和参考。项目实战瑞吉外卖的开发过程将帮助开发者熟悉Java开发技术,并理解实际项目的需求分析、系统设计和开发实施等流程。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值