Oracle根據連續性日期的重復數據取最大或最小值日期


原始數據:

 

 

 結果數據:

 

 

 對比兩個圖,要是不處理連續性中的重復值,我們直接可以用LEAD函數了事,但處理出來的結果貌似多余。

我的思路是先將原始數據中連續性日期有重復值的處理好,即選擇最小的一個,比如2011/01/01和2012/01/01的值是一樣,日期又是連續的,所以要去除2012/01/01,保留2011/01/01,讓處理出來的截止日期為第4行的日期-1即2012/10/30.

以下是實現代碼,有更好更方便的歡迎提出來,共同學習。

/*CREATE TABLE TB(PB_DATE DATE,SAVE_RATE VARCHAR2(10),LEN_RATE VARCHAR2(10));
INSERT INTO TB SELECT DATE'2010-01-01','8%','10%' FROM DUAL;
INSERT INTO TB SELECT DATE'2011-01-01','5%','7%' FROM DUAL;
INSERT INTO TB SELECT DATE'2012-01-01','5%','7%' FROM DUAL;
INSERT INTO TB SELECT DATE'2012-10-31','5%','6%' FROM DUAL;
INSERT INTO TB SELECT DATE'2013-01-01','5%','6%' FROM DUAL;
INSERT INTO TB SELECT DATE'2013-03-31','8%','9%' FROM DUAL;
INSERT INTO TB SELECT DATE'2013-09-01','8%','10%' FROM DUAL;
INSERT INTO TB SELECT DATE'2014-01-01','8%','9%' FROM DUAL;
INSERT INTO TB SELECT DATE'2015-01-01','6%','9%' FROM DUAL;
SELECT MIN(PB_DATE)START_DATE,MAX(END_DATE)END_DATE,SAVE_RATE,LEN_RATE FROM(
SELECT PB_DATE,LEAD(PB_DATE,1,DATE'9999-12-31')OVER(ORDER BY PB_DATE)-1 AS END_DATE,SAVE_RATE,LEN_RATE FROM TB)
GROUP BY SAVE_RATE,LEN_RATE;
SELECT T.*,MAX(PB_DATE)KEEP(DENSE_RANK LAST ORDER BY SAVE_RATE,LEN_RATE)OVER(PARTITION BY SAVE_RATE,LEN_RATE)AS RN FROM TB T ORDER BY 1 ;
SELECT * FROM TB;*/
/* Formatted on 2019/9/24 12:29:14 (QP5 v5.227.12220.39754) */
/* Formatted on 2019/9/24 12:41:16 (QP5 v5.227.12220.39754) */

實現代碼,以上代碼是造數腳本:
WITH TMP
AS ( SELECT PB_DATE,
SAVE_RATE,
LEN_RATE,
LEAD (PB_DATE, 1, DATE '9999-12-31') OVER (ORDER BY PB_DATE)
NEXT_DATE,
LAG (PB_DATE, 1, PB_DATE) OVER (ORDER BY PB_DATE) PRE_DATE,
LEAD (SAVE_RATE) OVER (ORDER BY PB_DATE) NEXT_SAVE,
LAG (SAVE_RATE) OVER (ORDER BY PB_DATE) PRE_SAVE,
LEAD (LEN_RATE) OVER (ORDER BY PB_DATE) NEXT_LEN,
LAG (LEN_RATE) OVER (ORDER BY PB_DATE) PRE_LEN
FROM TB
ORDER BY PB_DATE, SAVE_RATE, LEN_RATE),
TMP2
AS ( SELECT MAX (T.PB_DATE) AS PB_DATE, T.SAVE_RATE, T.LEN_RATE
FROM TMP T
WHERE (CASE
WHEN (SAVE_RATE = NEXT_SAVE AND LEN_RATE = NEXT_LEN)
OR (SAVE_RATE = PRE_SAVE AND LEN_RATE = PRE_LEN)
THEN
1
ELSE
0
END) = 1
GROUP BY T.SAVE_RATE, T.LEN_RATE
ORDER BY 1),
TMP3
AS (SELECT PB_DATE,SAVE_RATE,LEN_RATE
FROM TB T
WHERE NOT EXISTS
(SELECT NULL
FROM TMP2 R
WHERE R.SAVE_RATE = T.SAVE_RATE
AND R.LEN_RATE = T.LEN_RATE
AND R.PB_DATE = T.PB_DATE))
SELECT PB_DATE AS START_DATE,
LEAD (PB_DATE, 1, DATE '9999-12-31') OVER (ORDER BY PB_DATE) - 1
AS END_DATE,
SAVE_RATE,
LEN_RATE
FROM TMP3;


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM