Окно аналитической функции Oracle, определяемое данными - PullRequest
2 голосов
/ 25 января 2010

У меня есть таблица, которая представляет построчный дамп данных, считанных из определенного формата текстового файла. Каждая строка может представлять собой строку «master» или «detail», указанную с помощью кода rec_type. Я хотел бы написать запрос, который получает «главные» строки вместе со связанными подробными строками. Я придумала кое-что, что делает работу, но это кажется немного хакерским, и меня интересуют лучшие способы, если таковые имеются.

CREATE TABLE mdtest
 (rec_seq  NUMBER        PRIMARY KEY
 ,rec_type VARCHAR2(3)   NOT NULL
 ,rec_data VARCHAR2(100) NOT NULL);

INSERT INTO mdtest VALUES (1, '100', 'Bill Jones');
INSERT INTO mdtest VALUES (2, '200', '20080115,100.25');
INSERT INTO mdtest VALUES (3, '100', 'John Smith');
INSERT INTO mdtest VALUES (4, '200', '20090701,80.95');
INSERT INTO mdtest VALUES (5, '200', '20091231,110.35');

Желаемый результат:

SEQ_EMP  EMP_NAME    SEQ_DATA  EMP_DATA
=======  ==========  ========  ===============
      1  Bill Jones         2  20080115,100.25
      3  John Smith         4  20090701,80.95
      3  John Smith         5  20091231,110.35

Предположения:

  • записи обрабатываются в последовательности rec_seq
  • первый тип записи - "100"
  • каждая запись "100" содержит 1 или более записей "200" после

Примечание: это для Oracle 9i, однако в этом году мы должны перейти на 11g R1.

Ответы [ 2 ]

2 голосов
/ 25 января 2010

Вот что у меня есть:

SELECT seq_emp 
      ,SUBSTR(emp_seq_name,10) emp_name 
      ,seq_data 
      ,emp_data 
FROM  (SELECT MAX(CASE WHEN rec_type = '100' THEN rec_seq END) 
              OVER (ORDER BY rec_seq 
                    ROWS BETWEEN UNBOUNDED PRECEDING 
                             AND CURRENT ROW) seq_emp 
             ,MAX(CASE 
                  WHEN rec_type = '100' 
                  THEN TO_CHAR(rec_seq,'fm00000000') || '|' || rec_data 
                  END) 
              OVER (ORDER BY rec_seq 
                    ROWS BETWEEN UNBOUNDED PRECEDING 
                             AND CURRENT ROW) emp_seq_name 
             ,rec_seq seq_data 
             ,rec_type 
             ,rec_data emp_data 
       FROM   mdtest) 
WHERE  rec_type = '200' 
ORDER BY seq_data; 

Как вы можете видеть, я использую аналитическую функцию отчетов MAX с окном, начиная с верхней части набора и заканчивая текущей строкой, чтобы получить соответствующую запись "100" для текущей записи "200"; затем во внешнем запросе я отбрасываю ненужные записи "100".

Чтобы получить emp_name, мне пришлось добавить rec_seq с данными, чтобы функция MAX по-прежнему выбирала правильную запись заголовка; затем во внешнем запросе я отключаю rec_seq.

Я играл с другими аналитическими функциями и синтаксисом, включая FIRST_VALUE и синтаксис KEEP, но ни один из них, кажется, не упрощает эту работу; сложность в том, что окно определяется значением rec_type, а не постоянным смещением.

1 голос
/ 25 января 2010

В стремлении к простоте, как вы думаете, стоит ли загружать каждый тип записи в отдельную таблицу импорта перед обработкой?

create table mdtest100 as select * from mdtest where rec_type = 100;

create table mdtest200 as select * from mdtest where rec_type = 200;

with mdtest_detail as
    (
    select
        (select max(m.rec_seq) from mdtest100 m 
         where m.rec_seq < r200.rec_seq) master_rec_seq,
        r200.* 
    from 
        mdtest200 r200
    )
select
    m.rec_seq seq_emp,
    m.rec_data emp_name,
    d.rec_seq seq_data,
    d.rec_data emp_data
from
    mdtest_detail d
        inner join mdtest100 m on m.rec_seq = d.master_rec_seq
order by
    seq_emp,
    seq_data;


    SEQ_EMP  EMP_NAME    SEQ_DATA   EMP_DATA          
    1        Bill Jones  2          20080115,100.25          
    3        John Smith  4          20090701,80.95          
    3        John Smith  5          20091231,110.35          

Это может привести к более приемлемому решению и позволит вам разобрать и проверить отдельное поле EMP_DATA, отделенное от комм.

Просто мысль - извините, если вы ищете только решение для аналитики.

...