2009-03-03 2 views
0

Как получить сотрудников, скажем, 5 последних строк Action_reason, которые находятся в эффективной датированной записи, без будущих строк, должны выбирать только текущие и исторические строки (дата вступления в силу < = sysdate). Могу ли я получить их в одну строку или это будет 5 строк для сотрудника?Работа с эффективными устаревшими записями

select emplid, effdt, action_reasons 
-- we have to build a logic here. 
-- Should we initialize 5 ACT variables to fetch rows into it? 
-- Please help 
from JOB 
where emplid = '12345' 
    and effdt <= sysdate. 
+0

Зачем вам нужен один ряд? Что вам нужно в строках для людей с 4 строками действия? 3, 2, 1, 0? Разве вы не были бы лучше с таблицей между 0 и 5 рядами на одного сотрудника - как это предусмотрено решением @ Quasssnoi? –

+0

0 и 5 строк будут отображать 5 последних строк сотрудника, но причина действия может быть одинаковой. Я хочу получить 5 последних причин действия; если они имеют только 2-3 измененных строки, то остальные два будут пустыми. И это для целей отчетности. – 2009-03-03 17:51:10

ответ

3
SELECT LTRIM(SYS_CONNECT_BY_PATH(emplid || ', ' || effdt || ', ' || action_reasons, ', '), ', ') 
FROM (
     SELECT 
     FROM (
      SELECT emplid, effdt, action_reasons, ROW_NUMBER() OVER (ORDER BY effdt) AS rn 
      FROM JOB 
      WHERE emplid= '12345' 
       AND effdt <= SYSDATE 
      ) 
     WHERE rn <= 5 
     ) 
WHERE CONNECT_BY_ISLEAF = 1 
START WITH 
    rn = 1 
CONNECT BY 
    rn = PRIOR rn + 1 
+0

Как получить сотрудника (предположим 5 действий) в одной строке от последних до последних действий с соответствующими effdts, но не последние 5 строк из задания – 2009-03-03 17:12:28

+0

Извините, я не совсем понимаю, чего вы хотите. Пожалуйста, разместите в своем вопросе некоторые примеры данных и желаемый результат. – Quassnoi

+0

Он хочет: emplid, effdt1, act1, effdt2, act2, effdt3, act3, ...; Конечно, это не очень хорошая идея. –

0
SELECT JOBXX.EMPLID,JOBXX.EFFDT,JOBXX.ACT1,JOBXX.ACT2,JOBXX.ACT3,JOBXX.ACT4,JOBXX.ACT5 
FROM 
    (SELECT SD.EMPLID, 
      SD.EFFDT, 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 1 THEN SD.A1 ELSE 0 END)),1,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 1 THEN SD.A1 ELSE 0 END)),3,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 1 THEN SD.A1 ELSE 0 END)),5,2)) 
      AS ACT1, 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 2 THEN SD.A1 ELSE 0 END)),1,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 2 THEN SD.A1 ELSE 0 END)),3,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 2 THEN SD.A1 ELSE 0 END)),5,2)) 
      AS ACT2, 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 3 THEN SD.A1 ELSE 0 END)),1,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 3 THEN SD.A1 ELSE 0 END)),3,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 3 THEN SD.A1 ELSE 0 END)),5,2)) 
      AS ACT3, 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 4 THEN SD.A1 ELSE 0 END)),1,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 4 THEN SD.A1 ELSE 0 END)),3,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 4 THEN SD.A1 ELSE 0 END)),5,2)) 
      AS ACT4, 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 5 THEN SD.A1 ELSE 0 END)),1,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 5 THEN SD.A1 ELSE 0 END)),3,2)) || 
      CHR(SUBSTR(TO_CHAR(SUM(CASE WHEN SD.R3 = 5 THEN SD.A1 ELSE 0 END)),5,2)) 
      AS ACT5 
    FROM (
     SELECT EMPLID,EFFDT,ACTION_REASON, 
       SUBSTR(ACTION_REASON,1,1), 
       SUBSTR(ACTION_REASON,2,1), 
       SUBSTR(ACTION_REASON,3,1), 
       TO_NUMBER(ASCII(SUBSTR(ACTION_REASON,1,1)) || 
       ASCII(SUBSTR(ACTION_REASON,2,1)) || 
       ASCII(SUBSTR(ACTION_REASON,3,1))) AS A1, 
       ROW_NUMBER() over(PARTITION BY EMPLID,EFFDT ORDER BY EFFDT desc,EFFSEQ desC) R3 
     FROM PS_JOB 
     WHERE action in ('ABC','XYZ') 
     and action_reason in ('123','456','789') 
     and emplid IN('12345','ABCDE') 
     AND effdt between '01-jan-2008' and '18-dec-2008' 
     ORDER BY EFFDT DESC, EFFSEQ DESC 
    ) SD       
    GROUP BY EMPLID , EFFDT    
) JOBXX 
0

Вы можете иметь данные так, как вы хотите. Если вы хотите, как пять строк, то вы могли бы использовать это:

select * from (
      select emplid, empl_rcd, effdt, action_reason 
        , rank() over (partition by emplid, empl_rcd 
           order by effdt desc, effseq desc) rank1 
       from ps_job 
       where emplid = '12345' 
       and effdt <= sysdate) 
where rank1 <= 5 

Если вы хотите, чтобы данные все на одной линии, а затем использовать LAG аналитическую функцию Oracle, таким образом:

select * from ( 
    select emplid, empl_rcd, effdt 
     , lag(effdt) over(partition by emplid, empl_rcd order by effdt, effseq) effdt_lag1 
     , lag(effdt, 2) over(partition by emplid, empl_rcd order by effdt, effseq) effdt_lag2 
     , lag(effdt, 3) over(partition by emplid, empl_rcd order by effdt, effseq) effdt_lag3 
     , lag(effdt, 4) over(partition by emplid, empl_rcd order by effdt, effseq) effdt_lag4 
     , action_reason 
     , lag(action_reason) over(partition by emplid, empl_rcd order by effdt, effseq) action_reason_lag1 
     , lag(action_reason, 2) over(partition by emplid, empl_rcd order by effdt, effseq) action_reason_lag2 
     , lag(action_reason, 3) over(partition by emplid, empl_rcd order by effdt, effseq) action_reason_lag3 
     , lag(action_reason, 4) over(partition by emplid, empl_rcd order by effdt, effseq) action_reason_lag4 
     from ps_job 
    where emplid = '12345') j 
where effdt = (
      select max(j1.effdt) from ps_job j1 
      where j1.emplid = j.emplid 
       and j1.empl_rcd = j.empl_rcd 
       and j1.effdt <= sysdate) 

Это дает последний 5 значений effdt и последних 5 значений причины действия. Если вам не нужно, оба вышеуказанных SQL могут быть обрезаны соответствующим образом.