Not-null属性引用持久值的null或transient值

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:我很清楚我可以用级联持续映射孩子,但这不是这里的想法.我只想检查这两个是否独立工作.

mut*_*mar 9

这里的问题是您使用设置的值持久保存父表.当它继续存在时,它需要必须持久化的子表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改变映射


Vla*_*cea 2

除了父项与子项存在 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)