ren*_*nke 11 java orm hibernate jpa hibernate-mapping
我正在尝试使用JPA1和Hibernate实现来持久保存两个不同的实体.代码如下所示:
父实体类
@Entity
@Table(name = "parent")
public class Parent implements Serializable {
{...}
private Child child;
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "child_id", nullable = "false")
public Child getChild() {
return child;
}
public void setChild(Child child) {
this.child = child;
}
Run Code Online (Sandbox Code Playgroud)
子实体类
@Entity
@Table(name = "child")
public class Child implements Serializable {
private Integer id;
@Id
@Column(name = "child_id")
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
}
Run Code Online (Sandbox Code Playgroud)
测试用例
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:META-INF/application.xml")
@Transactional
public class ParentTest extends TestCase {
@PersistenceContext
private EntityManager entityManager;
@Test
public void testSave() {
Child child = new Child();
child.setId(1);
Parent parent = new Parent();
parent.setChild(child);
entityManager.persist(parent.getChild());
entityManager.persist(parent); // throws the exception
}
}
Run Code Online (Sandbox Code Playgroud)
application.xml上的实体管理器和事务
<tx:annotation-driven transaction-manager="transactionManager" />
<jee:jndi-lookup id="dataSource" jndi-name="java:/jdbc/myds" expected-type="javax.sql.DataSource" />
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="packagesToScan" value="com.mypackage" />
<property name="dataSource" ref="dataSource" />
<property name="jpaVendorAdapter"›
<bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"/>
</property>
<property name="jpaProperties">
<props>
<prop key="hibernate.dialect>org.hibernate.dialect.Oracle10gDialect</prop>
</props>
</property>
</bean>
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
Run Code Online (Sandbox Code Playgroud)
当尝试插入父对象时,hibernate会抛出一个PropertyValueException,表示child为null或者是瞬态,即使在此操作之前创建并保留了child.奇怪的是,这只能在单元测试中失败,而在实际应用中,使用预先插入的子项,这可以正常工作.
PS:我很清楚我可以用级联持续映射孩子,但这不是这里的想法.我只想检查这两个是否独立工作.
这里的问题是您使用设置的值持久保存父表.当它继续存在时,它需要必须持久化的子表id,因为它是一个外键,因此它是一个非null属性引用一个空值.
@Test
public void testSave() {
Child child = new Child();
child.setId(1);
entityManager.persist(child);
Parent parent = new Parent();
parent.setChild(child);
entityManager.persist(parent);
}
Run Code Online (Sandbox Code Playgroud)
试试这个先保存孩子,然后再保留父.Else改变映射
除了父项与子项存在 FK 问题之外,持续的顺序也是导致问题的原因。
你的问题与冲洗有关。仅仅因为您指示 Hibernate 持久保存一个对象,并不意味着它会自动满足您的请求。持久请求进入操作队列,仅在刷新时实现。因此,您的第二个持久性只需找到父实体,而没有实际的“持久性”子实体。
您可以简单地修复您的代码,如下所示:
Child child = new Child();
child.setId(1);
entityManager.persist(parent.getChild());
entityManager.flush();
Parent parent = new Parent();
parent.setChild(child);
entityManager.persist(parent);
entityManager.flush;
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
32967 次 |
最近记录: |