使用Spring Data-JPA更新单向@OneToMany对象列表时的常见行为是什么?

11

我有一个对象,其中包含另一个对象的列表。 它的映射如下:

@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 (?, ?)

我的问题是:删除整个列表并将其全部重新添加到数据库中是否是正确的行为?我原本期望它只会添加新图像,保留其他图像。而实际上,它会根据产品ID删除所有图像,然后再将它们全部重新添加。

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

这是正常的吗?有没有避免此删除的方法?

谢谢


不要描述你的代码,直接发布它。 - JB Nizet
你是否将图像添加到产品的List<Image>中,并调用image.set(product)方法? - Quang Dat Pham
不是的。这是一个单向的一对多关系。这意味着产品知道图像,但图像不知道产品。 - Igor
@JBNizet,我更新了问题并添加了代码。 - Igor
@Igor,你已经添加了用于复制预期行为的代码,但没有添加意外行为。这将会很有帮助。 - The Gilbert Arenas Dagger
发生的行为是每次我添加新图片时,它会删除所有内容并重新添加列表(带有新的ID)。我原本期望它只是添加新图片而不删除其他图片。如果您查看我粘贴的日志,您会看到在添加之前,它会从products_images表中删除所有内容,然后添加2个项目。即使我只添加了一个。它会删除所有内容,添加现有的内容,然后再添加新的内容。 - Igor
1个回答

5
简而言之,这需要在列表中进行排序,例如按照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<>();

任何一种方法都可以消除针对products_imagesdelete和额外的insert操作。这是我所知道的关于这个问题的高级解释,可以在这里阅读:链接,虽然它讨论的是集合中的@Embeddable而不是@Entity元素。似乎Hibernate应该更容易识别集合中的各个带有ID的实体,但如果是未排序的List(或Hibernate模型中的PersistentBag),它就不能这样做。请注意,保留了HTML标签。

2
谢谢!那解决了我的问题。在 List<Image> 上的 @OrderColumn 仍然有一些奇怪的问题。它不再删除整个列表,但它尝试更新其他项的 ID。使用 Set 它完美地工作。 - Igor

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接