簡體   English   中英

SQL 查詢以獲取給定日期的有效數據

[英]SQL Query to Get Valid Data on Given Date

我有一個數據庫,可以隨着時間的推移跟蹤個人工資,如下表所示:

數據庫基於

我想查詢每個月的人(基於id)工資給output如下表結果查詢

我不知道要使用什么查詢,因為它需要在薪水數據庫中迭代以檢查特定日期的有效薪水是多少。 對此有什么想法嗎?

謝謝!

這里有所有示例數據。 如前所述,您需要一個有效起始日期...

WITH
-- your input ...
indata(id,datevaliduntil,salary) AS (
          SELECT 1001,DATE '9999-12-31', 5000
UNION ALL SELECT 1001,DATE '2020-08-31', 4000
UNION ALL SELECT 1001,DATE '2020-04-30', 3000
)
,
-- make it almost like a slowly changing dimension
-- table - ad a valid-from-date ...
scd AS (
  SELECT
    id
  , LAG(datevaliduntil,1,DATE '1900-01-01') OVER (
      PARTITION BY id ORDER BY datevaliduntil
    ) AS datevalidfrom
  , datevaliduntil
  , salary
  FROM indata
)
,
-- the months from the example ...
months(monthend) AS (
  SELECT 
    mon::DATE - 1 AS monthend
  FROM 
  GENERATE_SERIES(
    '2020-04-01'::DATE
  , '2021-03-01'::DATE
  , INTERVAL '1 MONTH'
  ) gs(mon)
)
SELECT
  monthend
, id
, salary
FROM scd
JOIN months ON monthend >  datevalidfrom
           AND monthend <= datevaliduntil
ORDER BY 1
;
-- out   monthend  |  id  | salary 
-- out ------------+------+--------
-- out  2020-03-31 | 1001 |   3000
-- out  2020-04-30 | 1001 |   3000
-- out  2020-05-31 | 1001 |   4000
-- out  2020-06-30 | 1001 |   4000
-- out  2020-07-31 | 1001 |   4000
-- out  2020-08-31 | 1001 |   4000
-- out  2020-09-30 | 1001 |   5000
-- out  2020-10-31 | 1001 |   5000
-- out  2020-11-30 | 1001 |   5000
-- out  2020-12-31 | 1001 |   5000
-- out  2021-01-31 | 1001 |   5000
-- out  2021-02-28 | 1001 |   5000

這是使用橫向連接的方便位置。 以下是該月的第一天而不是最后一天——因為這樣更容易生成:

select i.id, gs.mon, s.salary
from generate_series('2019-01-01'::date, '2020-12-01'::date, interval '1 month') gs(mon) cross join
    (select distinct id from salaries) i left join lateral
    (select s.salary
     from salaries s
     where s.id = i.id and s.datevaliduntil >= gs.mon
     order by s.datevaliduntil asc
     limit 1
    ) s;

當然,如果你想要最后一天,你可以從每個日期中減去 1 天。

我會使用橫向連接,但反過來:從表本身開始,將前一個日期與lag()一起使用,然后使用生成系列生成兩者之間的日期。 需要一些額外的邏輯來調整月末:

select x.date - interval '1 day' date, t.id, t.salary
from (
    select id, salary,
        datevaliduntil + interval '1 day' datevaliduntil, 
        lag(datevaliduntil, 1, datevaliduntil) 
            over(partition by id order by datevaliduntil) + interval '1 day' lag_datevaliduntil 
    from mytable t
) t
cross join lateral generate_series(
    t.lag_datevaliduntil, 
    least(t.datevaliduntil, '2021-03-01'),
    '1 month'
) x(date)

您可以使用generate_series的第二個參數中的文字日期來控制總體上限(在這里,您希望在 2021 年 3 月結束時停止)。

DB Fiddle 演示

date                |   id | salary
:------------------ | ---: | -----:
2020-04-30 00:00:00 | 1001 |   3000
2020-04-30 00:00:00 | 1001 |   4000
2020-05-31 00:00:00 | 1001 |   4000
2020-06-30 00:00:00 | 1001 |   4000
2020-07-31 00:00:00 | 1001 |   4000
2020-08-31 00:00:00 | 1001 |   4000
2020-08-31 00:00:00 | 1001 |   5000
2020-09-30 00:00:00 | 1001 |   5000
2020-10-31 00:00:00 | 1001 |   5000
2020-11-30 00:00:00 | 1001 |   5000
2020-12-31 00:00:00 | 1001 |   5000
2021-01-31 00:00:00 | 1001 |   5000
2021-02-28 00:00:00 | 1001 |   5000

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM