按日期分组和合并数据的数量以创建新列

时间:2017-11-20 19:12:19

标签: sql oracle date group-by

我是SQL的初学者,并希望按日期/数字对连接数据进行分组以创建新列。拉出之后,我的数据如下所示:

select
    items.A,
    items.B,
    items.C,
    items.NUM,
    items.E,
    source.QTY,
    dates.DATE
from original_metric source
inner join grocery items
    on items.id = source.item_id
inner join day_dates dates
    on dates.date_id = source.date_id;

    A        B     C   NUM   E    QTY    DATE       
Vegetable carrot  John  1  Main   14  26-APR-11
Vegetable carrot  John  1  Main   35  27-APR-11
Vegetable carrot  John  1  Main    2  21-SEP-15
Vegetable carrot  John  1  Main   11  23-APR-17
Vegetable carrot  John  1  Main   25  22-MAY-17
Vegetable carrot  John  1  Main   20  20-APR-18
Vegetable onion   John  2  Extra  23  02-AUG-16
Vegetable onion   John  2  Extra  32  07-AUG-16
Meat      pork    Jane  3  Main   10  02-AUG-16
Meat      pork    Jane  3  Main   60  19-JAN-17
Meat      pork    Jane  3  Main   12  25-DEC-17

我希望它看起来像:

   A         B      C  NUM   E    QTY         DATE           QTR      WEEK_FILL    CURRENT_FILL 
Vegetable carrot  John  1  Main   49  24-APR-11:30-APR-11  2011Q4   none         100%
Vegetable carrot  John  1  Main    2  20-SEP-15:15-SEP-15  2015Q2   none         100% 
Vegetable carrot  John  1  Main   11  23-APR-17:29-APR-17  2017Q4   22.4%        73.4%
Vegetable carrot  John  1  Main   25  21-MAY-17:27-MAY-17  2017Q4   51%          73.4% 
Vegetable carrot  John  1  Main   20  15-APR-18:21-APR-18  2018Q4   80%          80% 
Vegetable onion   John  2  Extra  55  31-JUL-16:07-AUG-16  2016Q1   none         100%
Meat      pork    Jane  3  Main   10  31-JUL-16:07-AUG-16  2016Q1   none         100%
Meat      pork    Jane  3  Main   60  15-JAN-17:21-JAN-17  2017Q3   none         100%
Meat      pork    Jane  3  Main   12  24-DEC-17:30-DEC-17  2017Q3   none         100%

我想按周将NUM组分组。然后根据DATE显示它所属的QTR。

然后,对于WEEK_FILL,我试图加入以下逻辑:QTY(在那一周)除以总和(上一年同一季度的所有数量)除以NUM。如果没有前一年,请再往前走,直到有匹配的季度。否则,如果没有前几年,则将其保留为“无”。

然后,对于CURRENT_FILL,我试图合并一个类似的逻辑: 总和(该季度的所有数量)除以总和(与上一年同季度的所有数量)除以NUM。如果没有前一年,请再往前走,直到有匹配的季度。否则,如果没有前几年,则将其保留为100%。

(供参考,财政日历Q1:JUN,JUL,AUG; Q2:SEP,OCT,NOV; Q3:DEC,JAN,FEB; Q4:MAR,APR,MAY)

1 个答案:

答案 0 :(得分:1)

尝试此查询

WITH dataCTE AS(
  SELECT
    q.*,
    DENSE_RANK()OVER(PARTITION BY q.NUM,q.Q ORDER BY q.Y) Y_ORDER
  FROM
    (
      SELECT
        d.*,
        EXTRACT(YEAR FROM DT) Y,
        -- Q1: JUN, JUL, AUG; Q2: SEP, OCT, NOV; Q3: DEC, JAN, FEB; Q4: MAR, APR, MAY
        DECODE(EXTRACT(MONTH FROM DT),6,1,7,1,8,1,9,2,10,2,11,2,12,3,1,3,2,3,3,4,4,4,5,4) Q
      FROM TEST_DATA d
    ) q
)    
SELECT
  d.*,
  Y||'Q'||Q QTR,
  NVL(TO_CHAR(100*d.QTY/
    (
      SELECT SUM(p.QTY)
      FROM dataCTE p
      WHERE p.Y_ORDER=d.Y_ORDER-1
        AND p.Q=d.Q
        AND p.NUM=d.NUM
    ),'9999.99'),'none') WEEK_FILL,
  NVL(100*SUM(d.QTY)OVER(PARTITION BY d.Y,d.Q ORDER BY d.DT RANGE BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING)/
    (
      SELECT SUM(p.QTY)
      FROM dataCTE p
      WHERE p.Y_ORDER=d.Y_ORDER-1
        AND p.Q=d.Q
        AND p.NUM=d.NUM
    ),100) CURRENT_FILL
FROM dataCTE d

第二个变种

WITH dataCTE AS(
  SELECT
    q.*,
    DENSE_RANK()OVER(PARTITION BY q.NUM,q.Q ORDER BY q.Y) Y_ORDER
  FROM
    (
      SELECT
        d.*,
        EXTRACT(YEAR FROM DT) Y,
        -- Q1: JUN, JUL, AUG; Q2: SEP, OCT, NOV; Q3: DEC, JAN, FEB; Q4: MAR, APR, MAY
        DECODE(EXTRACT(MONTH FROM DT),6,1,7,1,8,1,9,2,10,2,11,2,12,3,1,3,2,3,3,4,4,4,5,4) Q
      FROM TEST_DATA d
    ) q
)
SELECT
  d.*,
  d.Y||'Q'||d.Q QTR,
  NVL(TO_CHAR(100*d.QTY/p.PREV_QTY,'9999.99'),'none') WEEK_FILL,
  NVL(100*SUM(d.QTY)OVER(PARTITION BY d.Y,d.Q ORDER BY d.DT RANGE BETWEEN UNBOUNDED PRECEDING AND UNBOUNDED FOLLOWING)/
      p.PREV_QTY,100) CURRENT_FILL
FROM dataCTE d
LEFT JOIN
  (
    SELECT Y_ORDER,Q,NUM,SUM(QTY) PREV_QTY
    FROM dataCTE
    GROUP BY Y_ORDER,Q,NUM
  ) p
ON p.Y_ORDER=d.Y_ORDER-1 AND p.Q=d.Q AND p.NUM=d.NUM

TEST_DATA是您的查询

CREATE TABLE TEST_DATA(
  A varchar2(20),
  B varchar2(20),
  C varchar2(20),
  NUM number,
  E varchar2(20),
  QTY number,
  DT date
);

INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','carrot','John',1,'Main',14,'26-APR-11');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','carrot','John',1,'Main',35,'27-APR-11');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','carrot','John',1,'Main', 2,'21-SEP-15');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','carrot','John',1,'Main',11,'23-APR-17');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','carrot','John',1,'Main',25,'22-MAY-17');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','carrot','John',1,'Main',20,'20-APR-18');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','onion','John',2,'Extra',23,'02-AUG-16');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Vegetable','onion','John',2,'Extra',32,'07-AUG-16');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Meat','pork','Jane',3,'Main',10,'02-AUG-16');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Meat','pork','Jane',3,'Main',60,'19-JAN-17');
INSERT INTO TEST_DATA(A,B,C,NUM,E,QTY,DT)VALUES('Meat','pork','Jane',3,'Main',12,'25-DEC-17');

SQL小提琴: 1 - http://sqlfiddle.com/#!4/e7eaa/49 2 - http://sqlfiddle.com/#!4/e7eaa/51

但我认为80%的{​​{1}}是错误的,因为2018Q4