我正在使用 JPA 实现 EclipseLink 和 MySQL 服务器作为应用程序的持久层。我的问题是,一个已经持久化并且已经有 id 的对象通过与 CascadeType.All 的多对一关系被第二次插入。
这里是实体:
@Entity
@Table(name = "messages")
public class Message extends DbObject {
@ManyToOne(cascade = CascadeType.ALL)
@JoinColumn(name = "user_id")
public User author;
public String message = "";
public Timestamp timestamp;
// Getters and setters removed ....
}
@Entity
@Table(name = "users")
public class User extends DbObject {
private String name;
// Getters and setters removed ....
}
@MappedSuperclass
public abstract class DbObject {
@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE)
private long id;
public long getId() {
return id;
}
public void setId(long id) {
this.id = id;
}
}
当我运行以下代码时,“用户”表中有两个条目。但我只想将现有用户分配给消息对象,而不是创建新用户。
User user = getCurrentUser(); // The user object is already persistent
Message msg = new Message();
msg.setUser(user);
EntityManager em = createEntityManager();
em.getTransaction().begin();
em.persist(msg);
em.getTransaction().commit();
em.close();
最佳答案
用户与您用来持久化消息的entityManager分离,并且在与用户的关系上设置了级联持久化,JPA要求持久化也发生在您分离的用户上。选项是 A) 使用当前持久性读取它,以便对其进行管理。除了级联之外,持久是对托管实体的无操作。 B) 删除Message->user关系上的所有级联 C) 对消息实例使用合并而不是持久。合并仍将保留它,但会正确地将用户状态复制到托管用户实例中。
关于java - 使用 CascadeType.ALL 时,JPA 将同一对象保留两次,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19202080/