级联操作有下面6个类型。下面实践 :级联保存(CascadeType是PERSIST)。
public enum CascadeType {
ALL,
PERSIST,
MERGE,
REMOVE,
REFRESH,
/**
* Cascade detach operation
*
* @since Java Persistence 2.0
*
*/
DETACH
}
例子:车库实体 CarRoom 中含有Set<Car> cars。车实体。
测试分为2个情况:
1、1边设置级联保存,另一边不设置。
2、2边都设置级联保存。
一、只是one的一方设置级联。
1、创建实体。
id实体:
@MappedSuperclass
public class IdentifiableEntity<ID extends Serializable> implements Serializable {
private static final long serialVersionUID = 4425736509958177L;
@Id
@GeneratedValue
private ID id;
车库实体:
@Entity
public class CarRoom extends IdentifiableEntity<Long> {
private String name;
@OneToMany(mappedBy="room")
private Set<Car> cars = new HashSet<Car>();
车的实体:
@Entity
public class Car extends IdentifiableEntity<Long> {
private String name;
@ManyToOne(cascade = CascadeType.PERSIST,fetch= FetchType.LAZY)
@JoinColumn(name = "room_id")
private CarRoom room;
2、测试级联保存 。从one的一方保存many的一方:
@Test
public void testPersist(){
CarRoom room = new CarRoom();
room.setName("库房1");
Car car1 = new Car();
car1.setName("汽车car1");
car1.setRoom(room);
Car car2 = new Car();
car2.setName("马车car2");
car2.setRoom(room);
room.getCars().add(car1);
room.getCars().add(car2);
carRepository.save(car2);
/**
* 会报错:异常信息:detached entity passed to persist: com.test.entity.CarRoom
* car2中的room是游离态的的实体传递持久化。
*/
carRepository.save(car1);
}
结果:上面想保存car来保存CarRoom ,第一个保存操作 carRepository.save(car2);成功了,但是保存第二个carRepository.save(car1)会报错:
org.springframework.dao.InvalidDataAccessApiUsageException: detached entity passed to persist: com.test.entity.CarRoom; nested exception is org.hibernate.PersistentObjectException: detached entity passed to persist: com.test.entity.CarRoom
数据库中只保存了car2。
通过dubug知道,在save(car2)之后,级联保存了room,所以car里面的room已经有了id。当JPA保存save(car1)的时候,改变了room的id值,并且想把room保存为新的数据。所以报错。
3、测试级联保存 。从many的一方保存one的一方:
@Test
public void testPersist2(){
CarRoom room = new CarRoom();
room.setName("库房1");
Car car1 = new Car();
car1.setName("汽车car1");
car1.setRoom(room);
Car car2 = new Car();
car2.setName("马车car2");
car2.setRoom(room);
room.getCars().add(car1);
room.getCars().add(car2);
carRoomRepository.save(room);
}
运行后结果如下:
只保存了room没有级联保存car1和car2。这是因为CarRoom实体中没有设置级联保存car。
二、两边都设置级联。
修改CarRoom实体如下:
public class CarRoom extends IdentifiableEntity<Long> {
private String name;
@OneToMany(mappedBy="room",cascade = CascadeType.PERSIST)
private Set<Car> cars = new HashSet<Car>();
运行后上面的2个测试代码后,car1、car2、room都保存了。
三、如果不配置级联保存,保存的时候对顺序有要求,先保存CardRoom ,再保存Car,否则保存不进去。
也就是要先保存被引用的一方
总结:
我们使用级联保存的时候,最后把两边都配置为级联保存。
如果只是one的一方配置了级联保存。那连续保存2个one的一方,并且one引用的是同一个many,就会出错。或者说要保存一个one的一方,并且one的一方引用了一个已经持久化的many的对象就会出错。
本文的源码地址:https://git.oschina.net/wkcom/Spring-data-jpa.git