SpringDataJpa大坑——一对多级联修改问题

前言

寒假接手一个项目,甲方提出了这样一个功能------需要一个商品有多张图片。可以进行滑动观看。这个需求很简单,前端只要做一个轮播图,后端只要涉及一个商品下有多组照片即可(一对多关系)。

项目后端选型

框架springboot 数据库框架springDataJpa

想要了解什么springDataJpa,可以看这篇文章

bug重现

甲方在后台图片进行图片拖动改变位置,点击进行修改按钮,该商品拖动改变位置的图片依旧没有发生变化。根据排查,前端发过来请求参数,确实改变了图片的位置。因此只能是后端背锅 啦!

可以给大家看看之前我写的前一版的代码(update方法)

java 复制代码
    @PostMapping("/updateModel")
    @ApiOperation(value = "更新型号", notes = "更新型号信息")
    public String updateModel(@RequestBody Model model) throws Exception {
        // 获取目标型号
        Model byModelId = modelService.findByModelId(model.getModelId());
        List<Image> targetImgs = byModelId.getImgs();
        if (byModelId == null) {
            throw new Exception("noModel");
        }
        if (targetImgs != null) {
            List<Image> imgs = model.getImgs();
            if(!CollectionUtils.isEmpty(imgs)) {
                List<String> url = imgs.stream().map(Image::getUrl).collect(Collectors.toList());
                for (Image image : model.getImgs()) {
                    url.add(image.getUrl());
                    imageService.addImage(image);
                }
                int waterMark = 1;
                // 改变了,要用原图进行添加水印,而不是用已经水印图进行再一次重复水印
                waterMark = createWaterMarkWithRaw(model.getModelName(), url);
                if (waterMark != 1) {
                    throw new Exception("添加水印失败");
                }
            }
        }
        // 更新保存到数据库
        modelService.updateModel(model);
        return "success";
    }

我在网上苦苦搜寻,都是关于对多端实体进行注解的添加,而我这个项目的多端就是商品下多个图片List< Image > imgs

java 复制代码
   @OneToMany(fetch=FetchType.EAGER,cascade=CascadeType.ALL,orphanRemoval = true)
   private List<Image> imgs;

cascade=CascadeType.ALL,orphanRemoval = true这两个说白了就是说进行级联操作,会对对应的子实体也会进行相应的改变(删除、插入等)

然后并没有什么卵用,我一直给干到一点中,弥留之际,我发现了问题的关键所在!

解决方案

因为jpa的机制,是可以通过注解@OneMany将实体创建表,也会创建关系表

在我这个项目就是Model 、Image和Model_img三张表。我在前面不断地测试插入,发现Model_img表没有发生改变 ,关联关系依旧是原来的"配对"。所以!说明咱们得update方法并没有进行修改Model_img表里所维护的数据。

那么我们只要手动干预model_img表中的关系数据,不就可以了吗。这样查出来就是修改后关系的数据了!

提交后的代码

java 复制代码
    @PostMapping("/updateModel")
    @ApiOperation(value = "更新型号", notes = "更新型号信息")
    public String updateModel(@RequestBody Model model) throws Exception {
        // 获取目标型号
        Model byModelId = modelService.findByModelId(model.getModelId());
        if (byModelId == null) {
            throw new Exception("noModel");
        }
        List<Image> newImags = new ArrayList<>();
        List<Image> imgs = model.getImgs();
        if(!CollectionUtils.isEmpty(imgs)) {
            List<String> url = imgs.stream().map(Image::getUrl).collect(Collectors.toList());
            for (Image image : model.getImgs()) {
                url.add(image.getUrl());
                imageService.addImage(image);
                Image newImage = new Image();
                newImage.setUrl(image.getUrl());
                newImags.add(newImage);
            }
            int waterMark = 1;
            // 改变了,要用原图进行添加水印,而不是用已经水印图进行再一次重复水印
            waterMark = createWaterMarkWithRaw(model.getModelName(), url);
            if (waterMark != 1) {
                throw new Exception("添加水印失败");
            }
        }
        // 清理原先数据
        model.getImgs().clear();
        // 添加修改位置的图片数据
        model.getImgs().addAll(newImags);
        modelService.updateModel(model);
        return "success";
    }

ps:多端实体也是需要进行添加相应的注解!即

java 复制代码
   @OneToMany(fetch=FetchType.EAGER,cascade=CascadeType.ALL,orphanRemoval = true)
   private List<Image> imgs;

到此,bug解决!时间终止为1点半!赚钱不易【叹气】

相关推荐
微笑听雨3 分钟前
Java 设计模式之单例模式(详细解析)
java·后端
微笑听雨3 分钟前
【Drools】(二)基于业务需求动态生成 DRL 规则文件:事实与动作定义详解
java·后端
猫猫的小茶馆22 分钟前
【STM32】FreeRTOS 任务的删除(三)
java·linux·stm32·单片机·嵌入式硬件·mcu·51单片机
天天摸鱼的java工程师27 分钟前
🔧 MySQL 索引的设计原则有哪些?【原理 + 业务场景实战】
java·后端·面试
空影学Java44 分钟前
Day44 Java数组08 冒泡排序
java
追风少年浪子彦1 小时前
mybatis-plus实体类主键生成策略
java·数据库·spring·mybatis·mybatis-plus
GEM的左耳返1 小时前
Java面试实战:从基础到架构的全方位技术交锋
spring boot·微服务·云原生·java面试·技术解析·ai集成
创码小奇客2 小时前
Talos 使用全攻略:从基础到高阶,常见问题一网打尽
java·后端·架构
jackzhuoa2 小时前
java小白闯关记第一天(两个数相加)
java·算法·蓝桥杯·期末
Rover.x2 小时前
内存泄漏问题排查
java·linux·服务器·缓存