【问题标题】:How generate gap rows between dates in postgresql如何在 postgresql 中的日期之间生成间隔行
【发布时间】:2016-08-03 21:01:22
【问题描述】:

我有一张存储付款的表格。我需要合并付款日期和不付款日期。

所以我会有类似的东西:

DeudaId Date Value
1 2016-01-01 $100 <- This come from a table
1 2016-01-02 $0   <- This was calculated
1 2016-01-03 $0   <- This was calculated
1 2016-01-04 $100 <- This come from a table

我有这个必要的解决方案,但对我来说太慢了:

CREATE OR REPLACE FUNCTION build_dates2()
 RETURNS TABLE (id INTEGER, cobrador_id INTEGER, fecha DATE)
 LANGUAGE plpgsql
 IMMUTABLE STRICT
AS $function$DECLARE min_date DATE;
DECLARE
    r RECORD;
    fecha RECORD;
BEGIN
    for r in (SELECT * FROM recaudo_deuda) LOOP
        for fecha in (select toFecha(r.fecha) + s.a AS dates from generate_series(0, r.plazo) as s(a)) LOOP
            return query VALUES ( r.id, r.cobrador_id, fecha.dates);
        END LOOP;
    END LOOP;
END
$function$;


SELECT * from  build_dates2() 

我知道我可以创建另一个表并使用触发器存储数据。我想知道是否存在一种有效的方法来即时执行此操作。

我还尝试使用 recaudo_deuda 表的最小值/最大值生成日期列表,但是我看不到如何从中构建结果:

CREATE OR REPLACE FUNCTION public.build_dates()
 RETURNS TABLE(dates date)
 LANGUAGE plpgsql
 IMMUTABLE STRICT
AS $function$
DECLARE min_date DATE;
DECLARE dias INTEGER;
BEGIN
    SELECT min(fecha), extract(DAY from max(fecha) - min(fecha))
        from recaudo_deuda INTO min_date, dias;

  RETURN QUERY select min_date + s.a AS dates from generate_series(0,dias) as s(a);
END
$function$

【问题讨论】:

    标签: sql postgresql postgresql-9.4 gaps-in-data


    【解决方案1】:

    您可以使用一条 SQL 语句来执行此操作,如下所示:

    select d.deudaid, s.a as date, coalesce(d.value, 0) as value
    from (
       select min(fecha), max(fecha)
       from recaudo_deuda
    ) m (mi, mx)
      cross join lateral generate_series(m.mi, m.mx, interval '1' day) as s(a)
      left join recaudo_deuda d on d.fecha = s.a::date
    order by s.a;
    

    【讨论】:

    • 但是如果想要项目 recaudo_deuda.id,这样我会得到空值。我需要将记录与日期相关联,因此在间隙中我得到了一系列具有 recaudo_deuda.id、GenerateDate、NULLs 的行...
    • @mamcx:你可以使用coalesce()
    • 不是value的问题,是Id。我需要保留 ID(建立存在或不支付该债务的日期列表)。请参阅有关问题的示例。
    • 我授予这个是因为它接近问题中的示例代码。实际上,我需要做点别的事情,比如select id, fecha + (generate_series(0, 60)|| ' days')::INTERVAL as fecha from recaudo_deuda;
    【解决方案2】:

    有点疯狂的解决方案。

    简单示例:

    with t(x) as (values(1),(2),(5),(10),(11))
    select
      x,
      generate_series(x, coalesce(lead(x) over (order by x) - 1, x)) as x_gaps,
      x = generate_series(x, coalesce(lead(x) over (order by x) - 1, x)) as x_real
    from t;
    

    转换为您的情况可能是(不确定我的列名是否准确):

    select
      deudaid,
      generate_series(fecha, coalesce(lead(fecha) over (order by fecha) - '1d', fecha), '1d') as fecha,
      (fecha = generate_series(fecha, coalesce(lead(fecha) over (order by fecha) - '1d', fecha), '1d'))::int * value as value
    from
      recaudo_deuda;
    

    【讨论】:

    • 我设法将查询提炼为select id, fecha + (generate_series(0, 60)|| ' days')::INTERVAL as fecha from recaudo_deuda;,但运行速度太慢。所以,我需要重复 deuda.id 字段 60 次。
    • @mamcx 顺便说一句,表格中有多少行没有/有填充的空白?我可以猜到大部分时间都花在了数据检索上。如果是这样,那么您应该在客户端而不是服务器端填补空白。
    猜你喜欢
    • 2012-05-28
    • 1970-01-01
    • 1970-01-01
    • 2012-12-16
    • 2016-11-23
    • 2021-09-28
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多