代码之家  ›  专栏  ›  技术社区  ›  Igor

用spring数据jpa更新单向@onetomany对象列表时的常见行为是什么?

  •  11
  • Igor  · 技术社区  · 7 年前

    我有一个对象和另一个对象的列表。 它的映射方式如下:

    @Entity
    @Inheritance(strategy = InheritanceType.JOINED)
    @Table(name = "products")
    public class Product extends DateAudit {
        private static final long serialVersionUID = 1L;
    
        @Id
        @GeneratedValue(strategy = GenerationType.IDENTITY)
        private Long id;
    
        @NotBlank
        @Size(min = 3, max = 30)
        private String name;
    
        @NotBlank
        private String shortDescription;
    
        @NotBlank
        private String description;
    
        @NotNull
        private Double regularPrice;
    
        private Double promotionPrice;
    
        @NotNull
        @ManyToOne(fetch = FetchType.LAZY)
        @JoinColumn(name = "category_id", nullable = false)
        private Category category;
    
        @NotNull
        @ManyToOne(fetch = FetchType.EAGER)
        @JoinColumn(name = "store_id", nullable = false)
        private Store store;
    
        @Size(max = 20)
        private String sku;
    
        private Double weight;
    
        private Integer quantityInStock;
    
        @NotNull
        private Boolean notifyLowStock;
    
        @OneToMany(cascade = CascadeType.ALL)
        private List<Image> images = new ArrayList<Image>();
    

    在图像方面,这是马坪:

    @Entity
    @Table(name = "images")
    public class Image {
    
        @Id
        @GeneratedValue(strategy = GenerationType.IDENTITY)
        private Long id;
    
        @NotBlank
        private String url;
    

    发生的是: 1。我创建了我的产品对象并将其保存在数据库中。 2。我更新此产品对象的方法是稍后向其添加图像,如下所示:

    Product product = repository.findById(productId);
    Image image = new Image();
    image.setUrl(url);
    product.getImages().add(image);
    repository.save(product);
    

    每当我向产品添加新图像并保存它时,控制台上就会显示以下内容:

    当我添加第一个图像时:

    2018-07-27 22:46:47.367 DEBUG 8580 --- [nio-5000-exec-3] org.hibernate.SQL                        : insert into images (url) values (?)
    2018-07-27 22:46:48.307 DEBUG 8580 --- [nio-5000-exec-3] org.hibernate.SQL                        : insert into products_images (product_id, images_id) values (?, ?)
    

    当我再添加一个图像时:

    2018-07-27 22:47:09.955 DEBUG 8580 --- [nio-5000-exec-4] org.hibernate.SQL                        : delete from products_images where product_id=?
    2018-07-27 22:47:09.957 DEBUG 8580 --- [nio-5000-exec-4] org.hibernate.SQL                        : insert into products_images (product_id, images_id) values (?, ?)
    2018-07-27 22:47:09.958 DEBUG 8580 --- [nio-5000-exec-4] org.hibernate.SQL                        : insert into products_images (product_id, images_id) values (?, ?)
    

    当我添加第三个图像时:

    2018-07-27 22:47:32.314 DEBUG 8580 --- [nio-5000-exec-5] org.hibernate.SQL                        : delete from products_images where product_id=?
    2018-07-27 22:47:32.316 DEBUG 8580 --- [nio-5000-exec-5] org.hibernate.SQL                        : insert into products_images (product_id, images_id) values (?, ?)
    2018-07-27 22:47:32.318 DEBUG 8580 --- [nio-5000-exec-5] org.hibernate.SQL                        : insert into products_images (product_id, images_id) values (?, ?)
    2018-07-27 22:47:32.319 DEBUG 8580 --- [nio-5000-exec-5] org.hibernate.SQL                        : insert into products_images (product_id, images_id) values (?, ?)
    

    我的问题是: 删除整个列表并将其全部添加回数据库是正确的行为吗?我本来以为它会添加新的图像,把其他图像留在那里。相反,如果删除基于productID的所有图像,然后重新将其全部添加回去。

    我在更新产品之前立即检索它。 我检索产品,将新图像添加到列表中,然后调用保存方法。

    这是正常的吗?有没有办法避免这种删除?

    谢谢

    1 回复  |  直到 7 年前
        1
  •  5
  •   df778899    7 年前

    简而言之,这需要列表中的一个订单,例如 Image.url :

    @OneToMany(cascade = CascadeType.ALL)
    @OrderColumn(name = "url")
    private List<Image> images = new ArrayList<>();
    

    或者不用担心订单:

    @OneToMany(cascade = CascadeType.ALL)
    private Set<Image> images = new HashSet<>();
    

    其中任何一个都消除了 delete 额外的 insert 反对 products_images .

    我所知道的最高级的解释是 here 尽管这是在说 @Embeddable 而不是 @Entity 集合中的元素。似乎Hibernate在识别集合中的单个实体(带有ID)时应该少一些问题,但它不会针对未排序的 List (或) PersistentBag 在休眠模式中)。