Порядок вычислений в спящем режиме - PullRequest
0 голосов
/ 09 февраля 2010

У меня есть столбец Oracle XMLType, в котором хранятся строки для разных языков. Мне нужно построить критерии гибернации, которые заказывают в этом столбце. Для этого мне нужно извлечь значение с помощью функции Oracle. Этот критерий генерируется автоматически кодом, который я написал, но я не могу понять, как извлечь из него значение и порядок через API критериев. По сути, сгенерированный SQL должен выглядеть примерно так:

SELECT EXTRACTVALUE(title, '//value[@lang="EN"]') AS enTitle
FROM domain_object 
ORDER BY enTitle

Я на мгновение возился с проекциями, но, похоже, они выполняют второй выбор. Что, как я полагаю, заставит hibernate выбрать ВСЕ значения и в памяти отсортировать их на основе проекции? Это было бы очень нежелательно = \

Ответы [ 2 ]

2 голосов
/ 09 февраля 2010

Хорошо, я нашел решение. Не уверен, что это best , поэтому я ненадолго оставлю его открытым, если кто-то захочет дать лучший ответ / уточнить мое решение.

Что я сделал, так это продлил org.hibernate.criterion.Order:

package com.mycorp.common.hibernate;

import org.hibernate.Criteria;
import org.hibernate.HibernateException;
import org.hibernate.criterion.CriteriaQuery;
import org.hibernate.criterion.Order;
import org.hibernate.engine.SessionFactoryImplementor;

import com.mycorp.LocalizationUtil;

public class LocalStringOrder extends Order {
    private static final long serialVersionUID = 1L;

    private boolean ascending;
    private String  propName;

    public LocalStringOrder(String prop, boolean asc) {
        super(prop, asc);
        ascending    = asc;
        propName = prop;
    }

    public String toSqlString(Criteria criteria, CriteriaQuery criteriaQuery) throws HibernateException {
        String[] columns = criteriaQuery.getColumnsUsingProjection(criteria, propName);
        StringBuffer fragment = new StringBuffer();
        for ( int i=0; i<columns.length; i++ ) {
            SessionFactoryImplementor factory = criteriaQuery.getFactory();
            fragment.append( factory.getDialect().getLowercaseFunction() )
            .append('(');
            fragment.append("EXTRACTVALUE(");
            fragment.append( columns[i] );
            fragment.append(", '//value[@lang=\"" + 
                LocalizationUtil.getPreferedLanguage().name() + 
                "\"')");
            fragment.append(')');
            fragment.append( ascending ? " asc" : " desc" );
            if ( i<columns.length-1 ) fragment.append(", ");
        }
        return fragment.toString();
    }

    public static Order asc(String propertyName) {
        return new LocalStringOrder(propertyName, true);
    }


    public static Order desc(String propertyName) {
        return new LocalStringOrder(propertyName, false);
    }
}

Тогда нужно было просто сказать criteria.addOrder(LocalStringOrder.asc('prop')).

1 голос
/ 05 августа 2010

Другим общим решением является NativeSQLOrder, см. http://opensource.atlassian.com/projects/hibernate/browse/HHH-2381. Я не понимаю, почему эта функция еще не в Hibernate.

...