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つが独立して機能するかどうかを確認したいだけです。
ここでの問題は、値が設定された親テーブルを永続化していることです。永続化する場合は、外部キーであり、したがって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を保存してマッピングを変更するには、これを試してください
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;
これを試して:
@ManyToOne(fetch = FetchType.LAZY,optional=false)
@JoinColun(name = "child_id", nullable = "false")
public Child getChild() {
return child;
}
一部のフィールドはnot-null="true"
で設定されています。dbに保存する前に、その値を指定する必要があります。親テーブルと子テーブルのすべてのフィールドに対して、適切な値を持つすべてのnull以外のフィールドを設定します
テストで「not-nullプロパティがnullまたは一時的な値を参照しています」というエラーが表示されたが、アプリケーションを正常に実行していないときに同様の問題が発生しました。
hibernate.properties
ファイルに1行追加するだけでよいことがわかりました。
hibernate.check_nullability=false
何らかの理由で、このオプションは、アプリケーションの実行時にはデフォルトでfalseに設定されていましたが、テスト時にはtrueに設定されていました。 hibernate.properties
に追加しても、実行中のアプリケーションには影響しませんでしたが、テストは修正されました。
@ManyToOne(fetch = FetchType.LAZY,cascade=PERSIST, mappedBy="parent")
にpublic Child getChild()
を指定して、親オブジェクトのみを永続化してください。どちらも永続化されます。