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点半!赚钱不易【叹气】

相关推荐
kinlon.liu几秒前
零信任安全架构--持续验证
java·安全·安全架构·mfa·持续验证
王哲晓21 分钟前
Linux通过yum安装Docker
java·linux·docker
java66666888826 分钟前
如何在Java中实现高效的对象映射:Dozer与MapStruct的比较与优化
java·开发语言
Violet永存27 分钟前
源码分析:LinkedList
java·开发语言
执键行天涯27 分钟前
【经验帖】JAVA中同方法,两次调用Mybatis,一次更新,一次查询,同一事务,第一次修改对第二次的可见性如何
java·数据库·mybatis
Jarlen42 分钟前
将本地离线Jar包上传到Maven远程私库上,供项目编译使用
java·maven·jar
蓑 羽1 小时前
力扣438 找到字符串中所有字母异位词 Java版本
java·算法·leetcode
Reese_Cool1 小时前
【C语言二级考试】循环结构设计
android·java·c语言·开发语言
工业甲酰苯胺1 小时前
Spring Boot 整合 MyBatis 的详细步骤(两种方式)
spring boot·后端·mybatis