Car と Engine という 2 つのエンティティがあります (例の名前ですが、ロジックは同じです)。
@Entity
public class Car {
@Id
private Long id;
@OneToOne(mappedBy = "car", cascade = Cascade.ALL)
private Engine engine;
...
}
@Entity
public class Engine {
@Id
private Long id; // 1
@MapsId // 2
@OneToOne
@JoinColumn(name = "car_id") // join column is optional here
private Car car;
...
}
だから、私はする:
em.save(car); // successfully saved, data is in the database, but (see below)
TypedQuery<Engine> query = em.createQuery("select engine from Engine engine where engine.car = :car", Engine.class)
query.setParameter("car", car);
query.getResultList();
例外をスローします:
ERROR [main] (JDBCExceptionReporter.java:234) - No value specified for parameter 1.
WARN [main] (TestContextManager.java:409) - Caught exception while allowing TestExecutionListener [org.springframework.test.context.transaction.TransactionalTestExecutionListener@17e5cbd] to process 'after' execution for test: method , exception [org.springframework.orm.jpa.JpaSystemException: org.hibernate.exception.DataException: could not execute query; nested exception is javax.persistence.PersistenceException: org.hibernate.exception.DataException: could not execute query]
ただし、Engine エンティティを車のインスタンス自体に @Id のみを持つように変更すると (//1 を削除し、// 2 を @Id に変更)、機能します。
JPAのドキュメントによると、同じように動作するはずです(少なくとも私は予想していました)。
私の環境: PostgreSQL 9、Spring Framework 3.1、Hibernate 3.6.8.Final、Tomcat 7 (JPA サポートは Spring インストルメンテーションによって追加されます)。
更新:EclipseLinkで両方のマッピングを試しましたが、うまくいきました。したがって、問題はおそらく Hibernate のどこかにあります。Hibernateで強制的に動作させる方法はまだわかりません。