Какое общее поведение при обновлении однонаправленного списка @OneToMany объектов с Spring Data-JPA?

У меня есть объект со списком другого объекта. Он отображается следующим образом:

@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>();

На стороне изображения, что maaping:

@Entity
@Table(name = "images")
public class Image {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @NotBlank
    private String url;

Что происходит: 1. Я создаю объект Product и сохраняю его в базе данных. 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 и добавьте все обратно.

Я извлекаю продукт прямо перед его обновлением. Я извлекаю продукт, добавляю новое изображение в список, и я вызываю метод save.

Это нормально? Есть ли способ избежать этого удаления?

Спасибо

Ответы

Ответ 1

Короче говоря, для этого нужен либо порядок в списке, например, 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.

Ближайшее, что я знаю на высоком уровне объяснения этого здесь, хотя говорить о @Embeddable вместо @Entity элементов в коллекции. Похоже, что Hibernate должен иметь меньше проблем с идентификацией отдельных объектов (с идентификаторами) в коллекции, но это не делает это для несортированного List (или PersistentBag в модели Hibernate).