У меня есть класс Fruit, класс Banana, который расширяет Fruit и класс YellowBanana, который расширяет Банан. У YellowBanana есть поле под названием «вкусно». Если я пытаюсь получить доступ к «вкусно» в запросе на основе дженериков я получаю следующее исключение:Спящий режим не может найти поле в производном классе (предположительно дженерики)
Unable to locate Attribute with the the given name [tasty] on this ManagedType [Fruit]
Упрощенная версия кода заключается в следующем:
@MappedSuperclass
public abstract class Fruit implements Serializable
{
}
@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
public class Banana extends Fruit
{
}
@Entity
public class YellowBanana extends Banana
{
private boolean tasty = false;
}
public abstract class GenericDao<T extends Fruit>
{
public List<T> getObjects(Filter filter /* some criteria passed here */)
{
EntityManager entityManager = getEntityManager();
CriteriaBuilder builder = entityManager.getCriteriaBuilder();
CriteriaQuery<T> query = builder.createQuery(getPersistentClass());
Root<T> root = query.from(getPersistentClass());
List<Predicate> predicates = getFilterQueryRestrictions(root, builder, filter);
if (predicates != null && !predicates.isEmpty())
query.where(predicates.toArray(new Predicate[] {}));
return entityManager.createQuery(query).getResultList();
}
private Class<T> getPersistentClass()
{
return (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];
}
}
public class BananaDao extends GenericDao<Banana>
{
@Override
protected List<Predicate> getFilterQueryRestrictions(Root<Banana> root, CriteriaBuilder builder,
Filter filter)
{
List<Predicate> restrictions = new ArrayList<Predicate>();
if (filter.forceTasty)
// The exception is on the following line on root.get:
restrictions.add(builder.equal(root.get("tasty"), 1));
return restrictions;
}
}
Как вы можете см. мою цель состоит в том, чтобы иметь общий DAO, который выполняет все операции CRUD для объектов Fruit. Для производных я создаю фактические классы DAO (BananaDao в этом примере). getPersistentClass() возвращает фактический общий тип. Этот общий способ DAO работает прекрасно до сих пор со всеми операциями, пока я не передал производный экземпляр (YellowBanana в этом примере).
Конечно, это лишь краткий пример. Фрукты на самом деле являются базовым классом всех моих entites (их много). Как я могу заставить Hibernate найти карту для «вкусного» поля?
Это не решит проблему пользователя. – Naros