1

我有几个具有基本LongID 和其他属性的实体。这些实体与另一个实体具有一对多的关系,该实体为第一个实体保存不同的自定义(用户输入)翻译。基本上存在实体对,其中一个包含所有非翻译内容,另一个包含文本属性的多个翻译。

为了减少代码和注释的重复,我想为其中一对实体中的每个实体创建一个抽象类。对于多个翻译,我创建了一个这样的类:

@MappedSuperclass
public abstract class CustomTranslations
{
    @Id
    protected Long id;
    @Id
    protected String locale;
}

对于主要实体,我有这个:

@MappedSuperclass
public abstract class CustomTranslationsHolder<T extends CustomTranslations>
{
    @OneToMany(fetch=FetchType.EAGER)
    @JoinColumn(name="ID")
    @MapKey(name="locale")
    protected Map<String, T> translationsByLocale;
}

所以说其中一个实体对是 for Foo。我会有这个:

@Entity
@Table(name="FOO_TRANSLATIONS")
public class FooTranslations extends CustomTranslations
{
    private String title;
    private String description;
    ....
}

和这个:

@Entity
public class Foo extends CustomTranslationsHolder<FooTranslations>
{
    @Id
    private Long id;
    private String whatever;
    private Integer blah;
    ...
    public String getTitle(String locale)
    {
        return translationsByLocale.get(locale).getTitle();
    }
}

这一切都编译得很好,但是在服务器启动时我收到以下错误:

Exception Description: Neither the instance method or field named [locale] exists for the item class [class java.lang.Void], and therefore cannot be used to create a key for the Map.
at org.eclipse.persistence.exceptions.ValidationException.mapKeyNotDeclaredInItemClass(ValidationException.java:1332)
at org.eclipse.persistence.internal.queries.MapContainerPolicy.initializeKey(MapContainerPolicy.java:517)
at org.eclipse.persistence.internal.queries.MapContainerPolicy.getKeyType(MapContainerPolicy.java:438)
at org.eclipse.persistence.internal.jpa.metamodel.MapAttributeImpl.<init>(MapAttributeImpl.java:167)
at org.eclipse.persistence.internal.jpa.metamodel.ManagedTypeImpl.initialize(ManagedTypeImpl.java:1158)
at org.eclipse.persistence.internal.jpa.metamodel.MetamodelImpl.initialize(MetamodelImpl.java:459)
at org.eclipse.persistence.internal.jpa.metamodel.MetamodelImpl.<init>(MetamodelImpl.java:111)
at org.eclipse.persistence.internal.jpa.metamodel.MetamodelImpl.<init>(MetamodelImpl.java:130)
at org.eclipse.persistence.internal.jpa.EntityManagerSetupImpl.getMetamodel(EntityManagerSetupImpl.java:2566)
at org.eclipse.persistence.internal.jpa.EntityManagerFactoryDelegate.getMetamodel(EntityManagerFactoryDelegate.java:592)
at org.eclipse.persistence.internal.jpa.EntityManagerFactoryImpl.getMetamodel(EntityManagerFactoryImpl.java:506)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.invokeProxyMethod(AbstractEntityManagerFactoryBean.java:376)
at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean$ManagedEntityManagerFactoryInvocationHandler.invoke(AbstractEntityManagerFactoryBean.java:517)
at $Proxy8.getMetamodel(Unknown Source)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:176)
at $Proxy12.getMetamodel(Unknown Source)
at org.springframework.data.jpa.repository.support.JpaEntityInformationSupport.getMetadata(JpaEntityInformationSupport.java:60)
at org.springframework.data.jpa.repository.support.JpaRepositoryFactory.getEntityInformation(JpaRepositoryFactory.java:149)
at org.springframework.data.jpa.repository.support.JpaRepositoryFactory.getTargetRepository(JpaRepositoryFactory.java:87)
at org.springframework.data.jpa.repository.support.JpaRepositoryFactory.getTargetRepository(JpaRepositoryFactory.java:70)
at org.springframework.data.repository.core.support.RepositoryFactorySupport.getRepository(RepositoryFactorySupport.java:137)
at org.springframework.data.repository.core.support.RepositoryFactoryBeanSupport.getObject(RepositoryFactoryBeanSupport.java:125)
at org.springframework.data.repository.core.support.RepositoryFactoryBeanSupport.getObject(RepositoryFactoryBeanSupport.java:41)
at org.springframework.beans.factory.support.FactoryBeanRegistrySupport.doGetObjectFromFactoryBean(FactoryBeanRegistrySupport.java:142)
... 44 more

所以,听起来它可以在地图locale上找到属性。T似乎它应该知道那T是 a CustomTranslations,因此知道 上有一个locale字段CustomTranslations

这只是 EclipseLink 的问题吗?或者有没有办法我可以这样做?我很想知道 Hibernate 如何处理完全相同的代码。任何有关这方面的信息或建议将不胜感激。

4

3 回答 3

8

接受的答案是错误的。泛型在 JPA 或 EclipseLink 中似乎没有得到很好的支持(可能是因为它们使用字节码编织而不是反射)。无论类型擦除如何,这些信息都绝对可以在运行时使用反射获得。

的实际运行时类型被删除,但扩展T的事实没有。这是在编译时指定的,并存储在可从实例访问的类元数据中。TCustomTranslationsClass

您可以调用CustomTranslationsHolder.class.getTypeParameters()which 将返回一个数组,其中包含一个TypeVariable名为“T”的元素,您可以在该数组上调用getBounds()which 将返回"CustomTranslations"。因此可以证明地图中的值包含该locale属性。

我会提交错误和/或与其他 JPA 提供程序进行测试。

于 2012-08-11T22:36:54.547 回答
3

似乎它应该知道 T 是CustomTranslations

由于类型擦除,它不知道这一点。

http://docs.oracle.com/javase/tutorial/java/generics/erasure.html

Java 泛型 - 类型擦除 - 何时以及发生什么

于 2012-08-04T21:20:04.587 回答
0

JPA 规范说:

映射超类定义的持久关系必须是单向的。

这就是您不能将CustomTranslationsHolder定义为映射超类的原因,因为它与另一个映射超类具有 ManyToOne 关系。这个类可能是一个实体。

@Entity
public abstract class CustomTranslationsHolder<T extends CustomTranslations>
{
    @OneToMany(fetch=FetchType.EAGER)
    @JoinColumn(name="ID")
    @MapKey(name="locale")
    protected Map<String, T> translationsByLocale;
}

但在这种情况下,不幸的是,JPA 将无法在运行时提取 T 类型。

我建议审查您的设计。

于 2016-04-14T07:41:08.140 回答