Объявленные атрибуты метамодели работают нормально, НО наследуемые атрибуты метамодели имеют значение NULL. Зачем? - PullRequest
5 голосов
/ 02 ноября 2010

Я не могу запустить следующий тест: -

@Test
public void test() {
    EntityManager em = entityManagerFactory.createEntityManager();
    em.getTransaction().begin();

    CriteriaBuilder builder = em.getCriteriaBuilder();
    CriteriaQuery<Project> query = builder.createQuery(Project.class);

    Root<Project> project = query.from(Project.class);

    Path<String> name = project.get(Project_.name);
    Assert.assertNotNull(name);

    Path<EntityLifeCycleImpl> lifeCycle = project.get(Project_.lifeCycle); // problem is here, throws NullPointer
    Assert.assertNotNull(lifeCycle);
}

выдает исключение NullPointerException в строке project.get(Project_.lifeCycle). Почему?

java.lang.NullPointerException
    at org.hibernate.ejb.criteria.path.AbstractPathImpl.get(AbstractPathImpl.java:138)

PersistenceEntityBase.java

import org.hibernate.annotations.GenericGenerator;
        @MappedSuperclass
@Access(AccessType.PROPERTY)
public abstract class PersistentEntityBase {

protected String identifier;

protected EntityLifeCycleImpl lifeCycle = new EntityLifeCycleImpl();

protected PersistentEntityBase() {
}

@Id
@GeneratedValue(generator="generator") 
@GenericGenerator(name="generator", strategy="guid", parameters = {})
@Column(name="identifier")
public String getIdentifier() {
    return identifier;
}

public void setIdentifier(String identifier) {
    this.identifier = identifier;
}

@Embedded
public EntityLifeCycleImpl getLifeCycle() {
    return lifeCycle;
}
public void setLifeCycle(EntityLifeCycleImpl lifeCycle) {
    this.lifeCycle = lifeCycle;
}

}

Project.java

@Entity
@Table(name="project")
@Access(AccessType.PROPERTY)
public class Project extends PersistentEntityBase {

    private String name;

    protected Project() {
    }

    public Project(String name) {
        this();
        this.name = name;
    }

    @Column(name="name", nullable=false, unique=true)
    public String getName() {
        return name;
}
public void setName(String name) {
    this.name = name;
}

}

EntityLifeCycleImpl.java

@Embeddable
public class EntityLifeCycleImpl implements EntityLifeCycle {
    private String createdBy;
    private Date createdDate;
       @Column(name="created_by")
public String getCreatedBy() {
    return createdBy;
}
public void setCreatedBy(String createdBy) {

    this.createdBy = createdBy;
}

@Column(name="created_date")
public Date getCreatedDate() {
    return createdDate;
}
public void setCreatedDate(Date createdDate) {
    this.createdDate = createdDate;
}

}

PersistentEntityBase_.java (Генерируется с помощью Hibernate Metamodel Generator)

   @StaticMetamodel(PersistentEntityBase.class)
public abstract class PersistentEntityBase_ {
    public static volatile SingularAttribute<PersistentEntityBase, EntityLifeCycleImpl> lifeCycle;
        public static volatile SingularAttribute<PersistentEntityBase, String> identifier;
     }

Project_.java

    @StaticMetamodel(Project.class)
public abstract class Project_ extends PersistentEntityBase_ {
    public static volatile SingularAttribute<Project, String> name;
    }

EntityLifeCycleImpl_.java

   @StaticMetamodel(EntityLifeCycleImpl.class)
public abstract class EntityLifeCycleImpl_ {
    public static volatile SingularAttribute<EntityLifeCycleImpl, String> createdBy;
    public static volatile SingularAttribute<EntityLifeCycleImpl, Date> createdDate;
}

persistence.xml (только относительная часть)

<class>com.comp.timetracking.entity.PersistentEntityBase</class>
<class>com.comp.timetracking.entity.Project</class>

EDIT: Я использую hibernate-entitymanager.3.6.0.Final и hibernate-jpamodelgen.1.0.0.Final.

РЕДАКТИРОВАТЬ 2 @Pascal
Я думаю, что Hibernate EM 3.6.0.Final позволяет нам определять @Embedded аннотированное поле на уровне @Entity, но оно запрещает такое поле на уровне @MappedSuperclass. Что ты скажешь?

Поскольку я не вижу здесь «опции загрузки файла», я загрузил TestCase в свою учетную запись esnips. Загрузите maven-based-project и запустите SingularAttributeTest.java. И проверить вывод консоли

ERROR main metamodel.MetadataContext:413 - Unable to locate static metamodel field : timetracking.entity.Employee_#lifeCycle

Нажмите на ссылку " Download embedded-singular-attribute.zip ", чтобы скачать файл БЕЗ установки менеджера загрузки. (Если вы нажмете Download link with Green arrow, вам придется установить менеджер загрузок !!)

Ответы [ 3 ]

6 голосов
/ 02 ноября 2010

Я протестировал ваши классы и тестовый пример с EclipseLink (я удалил определенные части Hibernate) и тесты прошли.Но это действительно терпит неудачу с Hibernate 3.5.6.Выглядит как ошибка в Hibernate.

Кстати, вам не хватает аннотации Temporal в вашей Embeddable

@Column(name = "created_date")
@Temporal(TemporalType.DATE)
public Date getCreatedDate() {
    return createdDate;
}

И вы не должны объявлять PersistentEntityBase вpersistence.xml (сопоставленный суперкласс не является сущностью).

1 голос
/ 29 октября 2014

это не проблема.

Из документации Hibernate:

Если класс X расширяет другой класс S, где S - это наиболее производный управляемый класс (т. Е. Объект или сопоставленный объект)суперкласс) расширяется на X, тогда класс X_ должен расширять класс S_, где S_ - класс метамодели, созданный для S.

https://docs.jboss.org/hibernate/entitymanager/3.5/reference/en/html/metamodel.html

1 голос
/ 28 декабря 2011

У меня была та же проблема: во время развертывания jboss 7.0.2 сообщал о следующей ошибке: «Невозможно найти статическое поле метамодели X» где X - это поле, унаследованное от @MappedSuperclass и аннотированное @Embedded (тип X - это класс, аннотированный @Embeddable).

Я решил это, добавив классы, аннотированные @Embeddable и @MappedSuperclass, вместе с классами, аннотированными @Entity, в список классов, управляемых модулем персистентности. Это настраивается в файле "persistence.xml".

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...