使用 Spring Data-JPA 更新单向@OneToMany 对象列表时的常见行为是什么?
What the common behaviour when updating an unidirectional @OneToMany List of objects with 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>();
在图像方面,这是映射:
@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<>();
其中任何一个都消除了 delete
和额外的 insert
针对 products_images
。
据我所知,最接近对此的高级解释的是 here,尽管它谈论的是集合中的 @Embeddable
而不是 @Entity
元素。似乎 Hibernate 在识别集合中的单个实体(带有 id)时应该有更少的问题,但它不会为未排序的 List
(或 Hibernate 模型中的 PersistentBag
)执行此操作。
我有一个对象和另一个对象的列表。 它的映射如下:
@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<>();
其中任何一个都消除了 delete
和额外的 insert
针对 products_images
。
据我所知,最接近对此的高级解释的是 here,尽管它谈论的是集合中的 @Embeddable
而不是 @Entity
元素。似乎 Hibernate 在识别集合中的单个实体(带有 id)时应该有更少的问题,但它不会为未排序的 List
(或 Hibernate 模型中的 PersistentBag
)执行此操作。