2

我有一个表格,它表示从特定文本文件格式读取的数据的逐行转储。每行可以代表一个“主”或“详细”行,通过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。

4

2 回答 2

2

这是我到目前为止所拥有的:

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 报告分析函数,窗口从集合顶部开始向下到当前行,以获取当前“200”记录的相关“100”记录;然后在外部查询中我丢弃不需要的“100”记录。

为了获得 emp_name,我必须在 rec_seq 中附加数据,以便 MAX 函数仍然选择正确的标头记录;然后在外部查询中我将rec_seq 切掉。

我使用过其他分析函数和语法,包括 FIRST_VALUE 和 KEEP 语法,但这些似乎都没有让这项工作变得更简单;困难在于窗口是由 rec_type 的值定义的,而不是一些恒定的偏移量。

于 2010-01-25T08:04:24.950 回答
1

为了简单起见,您认为是否值得在处理之前将每种记录类型加载到单独的导入表中?

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          

这可能有助于提供更易于维护的解决方案,并且可以让您分别解析和验证 comm 分隔的 EMP_DATA 字段。

只是一个想法 - 抱歉,如果您只是在寻找分析解决方案。

于 2010-01-25T11:25:46.600 回答