已经多次询问过,但我找不到任何好的答案,所以我会再问一遍。
我有如下父母 - 子女单向关系:
@Entity
@Table(name = "PARENT")
public class Parent {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "ID")
private Long parentId;
@OneToMany(fetch = FetchType.EAGER, cascade = CascadeType.PERSIST)
@JoinTable(name = "CHILD", joinColumns = @JoinColumn(name = "parent_id"), inverseJoinColumns = @JoinColumn(name = "ID"))
private List<Child> children;
}
@Entity
@Table(name = "CHILD")
public class Child {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "ID")
private Long id;
@Column(name = "PARENT_ID")
private Long parentId;
//some other field
}
我创建父项的实例,为其分配子项列表并尝试保留它:
Parent p = new Parent();
List<Child> children = new ArrayList<Child>();
Child c = new Child();
children.add(c);
p.addChildren(children);
em.merge(p);
当代码运行时,我得到以下异常:
MySQLIntegrityConstraintViolationException:无法添加或更新 子行:外键约束失败 (
testdb
。child
,CONSTRAINTparent_fk
FOREIGN KEY(parent_id
)参考parent
(id
)ON 删除没有动作更新没有行动)
我假设这是因为在尝试插入孩子时父母没有完全插入。
如果我不将子项添加到父项,则父项插入就好了。 我尝试切换GeneratedValue生成策略,但这没有帮助。
任何想法如何插入父母&amp;孩子们在同一时间?
编辑:即使我先保留父母,我仍然会收到同样的错误。我确定这是因为没有在孩子中设置parent_id;子进程是默认构造的,因此parent_id被设置为0,因此不存在外键约束验证。
有没有办法让jpa自动设置分配给父实例的子节点的parent_id?
答案 0 :(得分:11)
您的关系不一定是双向的。这里的评论中有一些错误的信息。
您还说您已将字段“parentId”添加到Child实体中,因为您认为JPA需要“知道”父字段,以便它可以设置该值。问题不在于JPA根据您提供的注释而不了解该字段。问题是您提供了有关该字段的“太多”信息,但该信息并非内部一致。
将Parent中的字段和注释更改为:
@OneToMany(fetch = FetchType.EAGER, cascade = CascadeType.PERSIST)
@JoinColumn(name = "parent_id")
private List<Child> children;
然后完全从Child实体中删除“parentId”。 您之前已指定了JoinTable注释。但是,你想要的不是JoinTable。 JoinTable将创建另外的第三个表,以便将两个实体相互关联。你想要的只是一个JoinColumn。将JoinColumn批注添加到也使用OneToMany注释的字段后,您的JPA实现将知道您正在将CHK添加到CHILD表中。问题是JPA已经使用列parent_id定义了CHILD表。
认为你给它两个CHILD表函数和parent_id列的冲突定义。在一种情况下,您告诉JPA它是一个实体,而parent_id只是该实体中的一个值。另一方面,您告诉JPA您的CHILD表不是实体,而是用于在CHILD和PARENT表之间创建外键关系。问题是您的CHILD表已经存在。然后当你持久化你的实体时,你已经告诉它parent_id显式为null(未设置)但是你还告诉它你的parent_id应该被更新以设置对父表的外键引用。
我使用上面描述的更改修改了代码,我也称为“persist”而不是“merge”。
这导致3个SQL查询
insert into PARENT (ID) values (default)
insert into CHILD (ID) values (default)
update CHILD set parent_id=? where ID=?
这反映了你想要的完美。 PARENT条目已创建。创建CHILD条目,然后更新CHILD记录以正确设置外键。
如果您改为添加注释
@OneToMany(fetch = FetchType.EAGER, cascade = CascadeType.PERSIST)
@JoinColumn(name = "parent_id", nullable = false)
private List<Child> children;
然后它会在插入子项时运行以下查询
insert into CHILD (ID, parent_id) values (default, ?)
从而从一开始就设置你的FK属性。
答案 1 :(得分:1)
向父实体添加updatable = false解决了在子表上执行插入和更新的问题。但是,我不知道为什么会这样,事实上,我不认为我在做什么是正确的,因为这意味着如果必须的话我以后不能更新子表。
答案 2 :(得分:0)
我知道坚持让有孩子的新父母使用em.persists(...)
为我工作。
使用em.merge(...)
,我真的不知道,但听起来它应该可行,但显然你遇到了麻烦,因为你的JPA实现试图在父母之前坚持孩子。
也许检查一下这是否适合您:https://vnageswararao.wordpress.com/2011/10/06/persist-entities-with-parent-child-relationship-using-jpa/
我不知道这是否会在您的问题中发挥作用,但请注意em.merge(p);
将返回一个托管实体...... p
将保持无法管理,并且您的孩子与p
相关联。
A)尝试em.persists(...)
而不是em.merge(...)
如果你不能
B)您正在合并parent
...而您cascade
设置为CascadeType.PERSIST
。尝试将其更改为
cascade=CascadeType.ALL
cascade={CascadeType.PERSIST, CascadeType.MERGE}
我知道merge
会保留新创建的实体,并且应该表现为persists
,但这些是我最好的提示。
答案 3 :(得分:0)
您希望通过此代码实现的目标。
@Entity
@Table(name = "PARENT")
public class Parent {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "ID")
private Long parentId;
@OneToMany(cascade = CascadeType.PERSIST)
private List<Child> children;
}
@Entity
@Table(name = "CHILD")
public class Child {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "ID")
private Long id;
@ManyToOne
Parent parent;
}