Почему Hibernate HQL работает, а Criteria - нет? - PullRequest
1 голос
/ 01 ноября 2011

У меня есть простая модель, представляющая собой цепочку отношений «один ко многим»: Страна - <Город - <Улица. Таблицы отображаются как объекты и возвращаются как Карта. </p>

Следующий метод тестированиявыдаёт странные результаты:

public static void main(String[] args) {
    Session session = HibernateSessionFactory.getSession();

    List<Map<String, Object>> results = null;

    //Query using HQL and print results
    System.out.println("FROM HQL =====================");
    String hql = "from Street where City.Country.countryid = 1";        
    Query query = session.createQuery(hql); 
    results = query.list();
    for(Map<String, Object> row : results) {
        System.out.println(row);
    }

    //Query using Criteria and print results
    System.out.println("FROM CRITERIA ================");
    Criteria criteria = session.createCriteria("Street");
    criteria.add(Restrictions.eq("City.Country.countryid", 1));
    results = criteria.list();
    for(Map<String, Object> row : results) {
        System.out.println(row);
    }
}

Верхний блок, использующий HQL, работает как положено, но нижний блок падает:

Вывод:

FROM HQL =====================
{streetname=Mayfair, City=org.hibernate.proxy.map.MapProxy@2b12e7f7, $type$=Street, streetid=1}
{streetname=Park Lane, City=org.hibernate.proxy.map.MapProxy@2b12e7f7, $type$=Street, streetid=2}
{streetname=Bond Street, City=org.hibernate.proxy.map.MapProxy@663b1f38, $type$=Street, streetid=3}
{streetname=Old Kent Road, City=org.hibernate.proxy.map.MapProxy@663b1f38, $type$=Street, streetid=4}
FROM CRITERIA ================
Exception in thread "main" org.hibernate.QueryException: could not resolve property: City.Country.countryid of: Street
    at org.hibernate.persister.entity.AbstractPropertyMapping.propertyException(AbstractPropertyMapping.java:67)
    at org.hibernate.persister.entity.AbstractPropertyMapping.toColumns(AbstractPropertyMapping.java:82)
    at org.hibernate.persister.entity.BasicEntityPropertyMapping.toColumns(BasicEntityPropertyMapping.java:54)
    at org.hibernate.persister.entity.AbstractEntityPersister.toColumns(AbstractEntityPersister.java:1367)
    at org.hibernate.loader.criteria.CriteriaQueryTranslator.getColumns(CriteriaQueryTranslator.java:457)
    at org.hibernate.loader.criteria.CriteriaQueryTranslator.getColumnsUsingProjection(CriteriaQueryTranslator.java:417)
    at org.hibernate.criterion.SimpleExpression.toSqlString(SimpleExpression.java:68)
    at org.hibernate.loader.criteria.CriteriaQueryTranslator.getWhereCondition(CriteriaQueryTranslator.java:357)
    at org.hibernate.loader.criteria.CriteriaJoinWalker.<init>(CriteriaJoinWalker.java:113)
    at org.hibernate.loader.criteria.CriteriaJoinWalker.<init>(CriteriaJoinWalker.java:82)
    at org.hibernate.loader.criteria.CriteriaLoader.<init>(CriteriaLoader.java:91)
    at org.hibernate.impl.SessionImpl.list(SessionImpl.java:1578)
    at org.hibernate.impl.CriteriaImpl.list(CriteriaImpl.java:306)
    at com.bar.foo(Main.java:33)

Iне могу понять, почему HQL может разрешить City.Country.countryid, а критерии (ограничения) не могут.

Я что-то упускаю из виду?

1 Ответ

1 голос
/ 01 ноября 2011

Потому что вы использовали неправильный синтаксис для запроса hql.несколько ошибок:

  • Street.class вместо "Street"
  • Отсутствует псевдоним
  • используйте страну напрямую вместо country.id

Попробуйте это:

Criteria criteria = session.createCriteria(Street.class)
.createAlias("city", "ci")
.add(Restrictions.eq("ci.country", country))

Подробнее см. в спящем режиме .

...