web-dev-qa-db-ja.com

非nullプロパティは永続化された値のnullまたは一時的な値を参照します

Hibernate実装で、JPA1を使用して2つの異なるエンティティを永続化しようとしています。このためのコードは次のとおりです。

親エンティティクラス

@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;
}

子エンティティクラス

@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;
    }
}

テストケース

@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
    }
}

エンティティーマネージャーと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>

親オブジェクトを挿入しようとすると、hibernateはPropertyValueExceptionをスローし、この操作の前に子が作成されて永続化されていても、子がnullまたは一時的であることを通知します。奇妙なことに、これは単体テストでのみ失敗します。実際のアプリケーションでは、子が事前に挿入されているため、これは期待どおりに機能します。

PS:カスケード永続化を使用して子をマッピングできることはかなりわかっていますが、それはここでは考えていません。これら2つが独立して機能するかどうかを確認したいだけです。

12
renke

ここでの問題は、値が設定された親テーブルを永続化していることです。永続化する場合は、外部キーであり、したがってnullではないプロパティがnull値を参照するため、すでに永続化する必要がある子テーブルIDが必要です。

    @Test
    public void testSave() {
        Child child = new Child();
        child.setId(1);
        entityManager.persist(child); 

        Parent parent = new Parent();
        parent.setChild(child);

        entityManager.persist(parent); 
    }

最初に子を保存し、次に親の.Elseを保存してマッピングを変更するには、これを試してください

8
muthukumar

Parentで最初に@ManyToOneを使用しているため、1つの子オブジェクトに複数の親があることがわかりますが、そうではありません。

あなたのクラス構造に従って、親エンティティと子エンティティの間にOneToOneマッピングがあることを理解できます。

例外に関して、保存と更新と削除のマッピングを自動的に処理するために親と子の間のカスケードを使用していない理由がありますか?あなたがそれについて考えたことがない場合は、ここで指定されているように以下の設定を設定することで試してみることができます: Example Parent/Child-Hibernate

cascade = {CascadeType.ALL}

ただし、子と親の間のマッピングをManyToOneからOneToOneに変更することをお勧めします。

私にお知らせください。

親から子へのFKの問題のほかに、永続的な順序が問題の原因です。

あなたの問題は フラッシュに関連する です。 Hibernateにオブジェクトを永続化するように指示したからといって、それが自動的にリクエストを処理するという意味ではありません。永続的なリクエストはアクションキューに送られ、フラッシュ時にのみ具体化されます。したがって、2番目の永続化は、実際の「永続化された」子なしで、親エンティティを見つけるだけです。

次のようにコードを修正するだけです。

    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;
2
Vlad Mihalcea

これを試して:

@ManyToOne(fetch = FetchType.LAZY,optional=false)
@JoinColun(name = "child_id", nullable = "false")
public Child getChild() {
    return child;
}
1
Tkachuk_Evgen

一部のフィールドはnot-null="true"で設定されています。dbに保存する前に、その値を指定する必要があります。親テーブルと子テーブルのすべてのフィールドに対して、適切な値を持つすべてのnull以外のフィールドを設定します

0
xrcwrn

テストで「not-nullプロパティがnullまたは一時的な値を参照しています」というエラーが表示されたが、アプリケーションを正常に実行していないときに同様の問題が発生しました。

hibernate.propertiesファイルに1行追加するだけでよいことがわかりました。

hibernate.check_nullability=false

何らかの理由で、このオプションは、アプリケーションの実行時にはデフォルトでfalseに設定されていましたが、テスト時にはtrueに設定されていました。 hibernate.propertiesに追加しても、実行中のアプリケーションには影響しませんでしたが、テストは修正されました。

0
Cassidy Laidlaw

@ManyToOne(fetch = FetchType.LAZY,cascade=PERSIST, mappedBy="parent")public Child getChild()を指定して、親オブジェクトのみを永続化してください。どちらも永続化されます。

0
DS Stack