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

标签 java hibernate spring-data one-to-many

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

@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. 我创建我的 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 的图像,然后将其全部重新添加回来。

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

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

谢谢

最佳答案

简而言之,这需要列表中的一个顺序,例如通过 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

据我所知,最接近对此进行高级解释的是 here ,尽管它谈论的是 @Embeddable 而不是集合中的 @Entity 元素。似乎 Hibernate 在识别集合中的单个实体(带有 id)时应该有更少的问题,但它不会为 Hibernate 中未排序的 List(或 PersistentBag)执行此操作型号)。

关于java - 使用 Spring Data-JPA 更新单向@OneToMany 对象列表时的常见行为是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51568481/

相关文章:

java - 使用 REST 和 Javaconfig 在 Spring Security 中摘要验证

java - 计算沿相对对角线的点偏移

java - 如何在 spring 数据中使用 elemMatch 为嵌套对象从 mongodb 检索数据

Spring 数据休息 - _links

java - 如何将双值从jsp存储到spring mvc Controller

java - Spring Data JPA 投影从数据库中选择的字段

java - Eclipse 中的 ical4j 出现 ParserException

Java:XML 解析器

java - Hibernate注释@OnetoOne似乎并没有限制数据库中的多对一条目

java - 版本字段打破了持久层上的 orphanRemoval 行为