Форум программистов «Весельчак У»
  *
Добро пожаловать, Гость. Пожалуйста, войдите или зарегистрируйтесь.
Вам не пришло письмо с кодом активации?

  • Рекомендуем проверить настройки временной зоны в вашем профиле (страница "Внешний вид форума", пункт "Часовой пояс:").
  • У нас больше нет рассылок. Если вам приходят письма от наших бывших рассылок mail.ru и subscribe.ru, то знайте, что это не мы рассылаем.
   Начало  
Наши сайты
Помощь Поиск Календарь Почта Войти Регистрация  
 
Страниц: [1]   Вниз
  Печать  
Автор Тема: Oracle. Сортировка внутри строки. И еще всякие сортировки.  (Прочитано 12376 раз)
0 Пользователей и 1 Гость смотрят эту тему.
RXL
Технический
Администратор

ru
Offline Offline
Пол: Мужской

WWW
« : 18-06-2009 09:24 » new

Собственно, задача уже решена — просто хочу поделиться материалом, т.к. в русскоязычном инете нихрена нет.

И так, задача отсортировать символы внутри строки.
Код: (SQL)
-- Глобальный тип.
CREATE TYPE char_table_t IS TABLE OF CHAR(1);

CREATE OR REPLACE FUNCTION SORT_STRING_INSIDE(p_str VARCHAR2) RETURN VARCHAR2 DETERMINISTIC AS
        n NUMBER;
        s VARCHAR2(256);
        st char_table_t := char_table_t();
BEGIN
        FOR n IN 1..LENGTH(p_str)
        LOOP
                st.EXTEND;
                st(n) := SUBSTR(p_str, n, 1);
        END LOOP;

        SELECT CAST(MULTISET(SELECT * FROM TABLE(st) ORDER BY 1) AS char_table_t)
        INTO st
        FROM DUAL;
       
        FOR n IN st.FIRST..st.LAST  
        LOOP
                s := s || st(n);
        END LOOP;
       
        RETURN s;
END;

Тестим:
Код: (SQL)
SELECT SORT_STRING_INSIDE('6CD25A3478') str FROM DUAL;

str
----------
2345678ACD

Метод "SELECT CAST(MULTISET(..." также можно использовать для сортировки других типов, но тип коллекции должен быть объявлен глобально — с локальными типами Oracle работать не желает (и прямо об этом говорит).
Об этом тут: http://technology.amis.nl/blog/1217/sorting-plsql-collections-the-quite-simple-way-part-two-have-the-sql-engine-do-the-heavy-lifting
« Последнее редактирование: 18-06-2009 09:42 от RXL » Записан

... мы преодолеваем эту трудность без синтеза распределенных прототипов. (с) Жуков М.С.
PooH
Глобальный модератор

ru
Offline Offline
Пол: Мужской
... и можно без хлеба!


« Ответ #1 : 18-06-2009 12:04 » 

Цитата
SELECT CAST(MULTISET(SELECT * FROM TABLE(st) ORDER BY 1) AS char_table_t)

А так не будет работать?
SELECT column_value FROM TABLE(CAST(st AS char_table_t))
Записан

Удачного всем кодинга! -=x[PooH]x=-
RXL
Технический
Администратор

ru
Offline Offline
Пол: Мужской

WWW
« Ответ #2 : 18-06-2009 12:17 » 

PooH, фишка в том, что таблица источник и приемник - одна и таже. Видимо для того и используется вложенный запрос. Я то, намучившись с собственными решениями, запрос нашел по ссылке выше.
Скажем так: я вообще не понимаю этих наворотов с CAST и MULTISET - не понимаю, почему просто SELECT * FROM TABLE(st) не работает.
Записан

... мы преодолеваем эту трудность без синтеза распределенных прототипов. (с) Жуков М.С.
PooH
Глобальный модератор

ru
Offline Offline
Пол: Мужской
... и можно без хлеба!


« Ответ #3 : 18-06-2009 12:50 » 

У тебя вроде как получается...
SELECT * FROM TABLE(st) ORDER BY 1 овзвращает курсор
MULTISET - перегоняет курсор в множество (коллекцию?)
CAST - собственно кастует множество в "тип"
в результате получается единственная запись с типом данных "тип"
и успешно помещается в переменную.

теперь я понял, почему мной предложенный вариант работать не будет =)
Записан

Удачного всем кодинга! -=x[PooH]x=-
RXL
Технический
Администратор

ru
Offline Offline
Пол: Мужской

WWW
« Ответ #4 : 18-06-2009 13:11 » 

Ну, а я, после твоих объяснений, наконец-то понял, нафига все это делается. Улыбаюсь
Вообще, сперва я пытался определять типы внутри функции и видимо оттуда все мои неудачи. Но Оракл дает довольно бестолковые ошибки при компиляции и из них я не мог понять, в чем проблема. Только когда перешел к текущей форме запроса он ругнулся, что нужен внешний тип.
Записан

... мы преодолеваем эту трудность без синтеза распределенных прототипов. (с) Жуков М.С.
Страниц: [1]   Вверх
  Печать  
 

Powered by SMF 1.1.21 | SMF © 2015, Simple Machines