Hibernate: не удается использовать именованный параметр для OFFSET и LIMIT?

Я пытаюсь получить следующее NamedQuery работает:

@NamedQuery(name="MyEntity.findByUser", query="SELECT m FROM MyEntity m WHERE m.owner = :user OFFSET :offset LIMIT :limit")

проблема в том, что это заставляет Hibernate взорваться со следующей трассировкой стека при запуске сервера:

[INFO] [talledLocalContainer] java.lang.NullPointerException
[INFO] [talledLocalContainer]   at org.hibernate.hql.ast.ParameterTranslationsImpl.getNamedParameterExpectedType(ParameterTranslationsImpl.java:63)
[INFO] [talledLocalContainer]   at org.hibernate.engine.query.HQLQueryPlan.buildParameterMetadata(HQLQueryPlan.java:296)
[INFO] [talledLocalContainer]   at org.hibernate.engine.query.HQLQueryPlan.<init>(HQLQueryPlan.java:97)
[INFO] [talledLocalContainer]   at org.hibernate.engine.query.HQLQueryPlan.<init>(HQLQueryPlan.java:56)
[INFO] [talledLocalContainer]   at org.hibernate.engine.query.QueryPlanCache.getHQLQueryPlan(QueryPlanCache.java:72)
[INFO] [talledLocalContainer]   at org.hibernate.impl.SessionFactoryImpl.checkNamedQueries(SessionFactoryImpl.java:400)
[INFO] [talledLocalContainer]   at org.hibernate.impl.SessionFactoryImpl.<init>(SessionFactoryImpl.java:351)
[INFO] [talledLocalContainer]   at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:1291)
[INFO] [talledLocalContainer]   at org.hibernate.ejb.Ejb3Configuration.buildEntityManagerFactory(Ejb3Configuration.java:713)
[INFO] [talledLocalContainer]   at org.hibernate.ejb.HibernatePersistence.createEntityManagerFactory(HibernatePersistence.java:121)
[INFO] [talledLocalContainer]   at javax.persistence.Persistence.createEntityManagerFactory(Persistence.java:83)
[INFO] [talledLocalContainer]   at javax.persistence.Persistence.createEntityManagerFactory(Persistence.java:60)
(...)

после некоторых проб и ошибок я обнаружил, что замена ": offset" и": limit " литеральными значениями (0 и 10, соответственно) работала нормально. Есть причина для этого, и есть ли способ, чтобы получить именованные параметры в запросе?

Я видел другие примеры это использует позиционированные параметры для динамического задания смещения и предельных значений в именованном запросе, но я бы не хотел, чтобы мой код вырождался в кучу нечитаемых query.setParameter(1, "someValue"); бред. Именованные параметры должны были избавиться от такого рода кода мусора.

1 ответов


Hibernate имеет специальный API для указания этих понятий во время выполнения. Попробуйте это:

@NamedQuery(name="MyEntity.findByUser", 
    query="SELECT m FROM MyEntity m WHERE m.owner = :user") // don't put OFFSET or LIMIT here

...

entityManager.createNamedQuery("MyEntity.findByUser") 
.setFirstResult(20) // equivalent to OFFSET
.setMaxResults(5) // equivalent to LIMIT
.getResultList();

Я думаю, что причина, по которой это делается таким образом, заключается в том, что поставщики баз данных сильно различаются относительно того, как и где в SQL-запросе указаны эти понятия, поэтому не разумно выбирать один формат над другим и слишком сложно пытаться конвертировать между ними.

таким образом, реализация диалекта четко знает, что нужно сделать, и может это сделать.