首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在JPA中映射Map<Entity,Entity>结果在表中不存在于此描述符中

在JPA中映射Map<Entity,Entity>结果在表中不存在于此描述符中
EN

Stack Overflow用户
提问于 2018-01-11 19:31:40
回答 2查看 997关注 0票数 1

我需要映射一个Java,其中键和值都是JPA实体。我在一本JPA 2书(Pro JPA 2)中读到,这应该用@OneToMany@ManyToMany注释来完成,而不是ElementCollection,因为两者都是JPA实体。

我可以让@ManyToMany注释在没有问题、没有属性或任何东西的情况下工作。但据我所知,这种关系是一对多的。对于任何值实体,只能有一个类的实体保存映射(我在这里正确地理解了@OneToMany,因为它与地图有关吗?)

简单地用@OneToMany注释地图,如下所示:

代码语言:javascript
复制
@OneToMany
private Map<Stage, ScoreCard> scoreCards;

这一例外的结果是:

厄尔尼诺: 2018-01-11 21:11:10.755--ServerSession(53937593)--Exception EclipseLink-0:org.eclipse.persistence.exceptions.IntegrityException描述符 例外情况: 异常说明: org.eclipse.persistence.exceptions.DescriptorException异常描述:表记分卡不在此描述符中。描述符: RelationalDescriptor(fi.ipsc_result_server.domain.ResultData.CompetitorResultData ->DatabaseTable(COMPETITORRESULTDATA) org.eclipse.persistence.exceptions.DescriptorException异常描述:必须为序列号字段定义非只读映射。描述符: RelationalDescriptor(fi.ipsc_result_server.domain.ResultData.CompetitorResultData ->DatabaseTable(COMPETITORRESULTDATA)

我已经找到了许多关于Map的JPA注释的例子,但是没有一个能说明如何用实体作为键和值对地图进行注释。

这两个类都使用@Entity进行注释,并且都有一个Id,并且可以在没有问题的情况下持久化,除非试图获得这个Map持久化,所以问题在于地图的注释。

作为测试,我编写了一个包含Map: package fi.ipsc_result_server.domain.ResultData的测试类;

代码语言:javascript
复制
@Entity
public class TestClass {

    @Id
    @GeneratedValue(strategy = GenerationType.SEQUENCE)
    Long id;

    @OneToMany
    @MapKeyJoinColumn(name="testKey")
    Map<TestKeyClass, TestValueClass> testMap;

[getters and setters]
}
package fi.ipsc_result_server.domain.ResultData;
@Entity
public class TestKeyClass {
    @Id
    @GeneratedValue(strategy = GenerationType.SEQUENCE)
    Long id;

[getter and setter for id]
}

package fi.ipsc_result_server.domain.ResultData;

@Entity
public class TestValueClass {
    @Id
    @GeneratedValue(strategy = GenerationType.SEQUENCE)
    Long id;

[getter and setter for id]
}

这导致了同样的例外情况:

描述符异常: 异常描述: org.eclipse.persistence.exceptions.DescriptorException异常描述:表TESTVALUECLASS不存在于此描述符中。描述符: RelationalDescriptor(fi.ipsc_result_server.domain.ResultData.TestClass ->DatabaseTable(TESTCLASS) org.eclipse.persistence.exceptions.DescriptorException异常描述:必须为序列号字段定义非只读映射。描述符: RelationalDescriptor(fi.ipsc_result_server.domain.ResultData.TestClass ->DatabaseTable(TESTCLASS)

EN

回答 2

Stack Overflow用户

发布于 2018-01-11 19:38:22

Map<Entity,Entity>可以与@OneToMany@ManyToMany一起使用。您只需添加一个附加注释:

代码语言:javascript
复制
@OneToMany
@MapKeyJoinColumn(name="stage")
private Map<Stage, ScoreCard> scoreCards;

其名称为:

映射键的外键列的名称。

更新

您需要确保value实体具有对键实体的引用:

代码语言:javascript
复制
public clas ScoreCard{

  @ManyToOne
  private Stage stage;
}
票数 0
EN

Stack Overflow用户

发布于 2018-01-11 21:30:31

因此,我通过在Map上的注释中添加@MapKeyJoinColumn而不是@MapKeyJoinColumn来实现这一点。在我在问题结束时包含的TestClass中,这是可行的,创建了表,我可以持久化TestClass的一个实例,一旦我持久化了TestKeyClass实例,因为级联显然不包括关键的值实体:

代码语言:javascript
复制
@OneToMany(cascade = CascadeType.PERSIST)
@JoinColumn(name="TESTCOLUMN")
Map<TestKeyClass, TestValueClass> testMap;

所以。我不知道这是为什么工作,为什么需要它,为什么@MapKeyJoinColumn不能工作。上面的注释生成一个TESTVALUECLASS表,其中列为"TESTCOLUMN“(来自@JoinColumn注释)和"testMap_KEY”列。列"TESTCOLUMN“是TestClass的联接列,具有TestClass实体的id (具有映射的类)。"testMap_KEY“是作为特定值的键的TestKeyClass实例的id。因此,出于某种原因,EclipseLink在没有@JoinColumn注释的情况下不会生成所需的列。

在某种程度上,我通过阅读一些JPA,并意识到在我当前的项目中我会很好地掌握JPA,从而刷新了我对JPA的贫乏知识。和以前一样,我和以前一样困惑于JPA的一些方面,我只是觉得“哦,好吧,不管怎么样,它都管用”.哦,好吧!谢谢你的帮助,马西杰!

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/48214554

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档