如果当前为空 Oracle,则插入前一个日期的信息

Inserting the previous date's info if the current is empty Oracle

我有一个 table 包含以下信息

Table1 是一个子查询,为了简化我将只使用结果 table:

| Account_No | Dept_ID | Currency| Amount | Date_2(dd/mm/yyyy)|
+------------+---------+---------+--------+-------------------+
| 1          | 1       | USD     | 50     | 03/01/2017        |
| 1          | 2       | EUR     | 25     | 01/01/2017        |
| 1          | 3       | USD     | 51     | 01/01/2017        |
| 1          | 1       | GBP     | 45     | 01/01/2017        |
| 1          | 2       | USD     | 65     | 02/01/2017        |

Amount是某部门指定币种账户在该日的日终金额。更重要的是,同一账户可以有 CurrencyDept_ID 的变化(相同 Account_no 但货币不同and/or department ID) 我的意思是 PK 是 Account_no, Dept_ID 和 Currency.

的组合

我正在尝试将 table 附加到日期 table 中,该日期每天都在某个指定范围内:

日期 table:

|   Date_1   |
+------------+
| 01/01/2017 |
| 02/01/2017 |
| 03/01/2017 |
| 04/01/2017 |
| 05/01/2017 |
...

预期结果是:

|   Date_1   | Account_No | Dept_ID | Currency| Amount | Date_2(dd/mm/yyyy)|
+------------+------------+---------+---------+--------+-------------------+
| 01/01/2017 | 1          | 1       | USD     | 0      |                   |
| 01/01/2017 | 1          | 2       | USD     | 0      |                   |
| 01/01/2017 | 1          | 2       | EUR     | 25     | 01/01/2017        |
| 01/01/2017 | 1          | 3       | USD     | 51     | 01/01/2017        |
| 01/01/2017 | 1          | 1       | GBP     | 45     | 01/01/2017        |
| 02/01/2017 | 1          | 2       | USD     | 65     | 02/01/2017        |
| 02/01/2017 | 1          | 2       | EUR     | 25     |                   |
| 02/01/2017 | 1          | 3       | USD     | 51     |                   |
| 02/01/2017 | 1          | 1       | GBP     | 45     |                   |
| 02/01/2017 | 1          | 1       | USD     | 0      |                   |
| 03/01/2017 | 1          | 1       | USD     | 50     | 03/01/2017        |
| 03/01/2017 | 1          | 1       | GBP     | 45     |                   |
| 03/01/2017 | 1          | 3       | USD     | 51     |                   |
| 03/01/2017 | 1          | 2       | EUR     | 25     |                   |
| 03/01/2017 | 1          | 2       | USD     | 65     |                   |

因此,对于 Dates table 中的每个日期,我将从 Table1 中获取信息,如果信息丢失,则它应该 select 前几天的信息。我已经完成了左连接的查询,但不知道如何将前一天的数据填充到缺失的字段中

SELECT * FROM DATES A LEFT JOIN TABLE1 B ON A.DATE_1 = B.DATE_2;

我明白了

|   Date_1   | Account_No | Dept_ID | Currency| Amount | Date_2(dd/mm/yyyy)|
+------------+------------+---------+---------+--------+-------------------+
| 01/01/2017 | 1          | 2       | EUR     | 25     | 01/01/2017        |
| 01/01/2017 | 1          | 3       | USD     | 51     | 01/01/2017        |
| 01/01/2017 | 1          | 1       | GBP     | 45     | 01/01/2017        |
| 02/01/2017 | 1          | 2       | USD     | 65     | 02/01/2017        |
| 03/01/2017 | 1          | 1       | USD     | 50     | 03/01/2017        |
| 04/01/2017 |            |         |         |        |                   |
...

欢迎提供有关如何进行的建议

您可以使用 lag()ignore nulls 选项来完成此操作。我想这就是你想要的:

select d.date_1, a.account_no,
       coalesce(dept_id,
                lag(dept_id ignore nulls) over (partition by t1.account_no order by d.date_1)
               ) as dept_id,
       coalesce(currency,
                lag(currency ignore nulls) over (partition by t1.account_no order by d.date_1)
               ) as currency,
       coalesce(amount,
                lag(amount ignore nulls) over (partition by t1.account_no order by d.date_1)
               ) as amount
from dates d CROSS JOIN
     (select distinct account_no from table1) a left join
     table1 t1
     on d.DATE_1 = t1.DATE_2 and a.account_no = t1.account_no;

您还可以使用 partition outer join 加上一个 case 语句来决定是放置当前数量还是之前可用的数量(如果没有当前数量),如下所示:

WITH table1 AS (SELECT 1 account_no, 1 dept_id, 'USD' currency, 50 amount, to_date('03/01/2017', 'dd/mm/yyyy') date_2 FROM dual UNION ALL
                SELECT 1 account_no, 2 dept_id, 'EUR' currency, 25 amount, to_date('01/01/2017', 'dd/mm/yyyy') date_2 FROM dual UNION ALL
                SELECT 1 account_no, 3 dept_id, 'USD' currency, 51 amount, to_date('01/01/2017', 'dd/mm/yyyy') date_2 FROM dual UNION ALL
                SELECT 1 account_no, 1 dept_id, 'GBP' currency, 45 amount, to_date('01/01/2017', 'dd/mm/yyyy') date_2 FROM dual UNION ALL
                SELECT 1 account_no, 2 dept_id, 'USD' currency, 65 amount, to_date('02/01/2017', 'dd/mm/yyyy') date_2 FROM dual),
      dates AS (SELECT to_date('01/01/2017', 'dd/mm/yyyy') date_1 FROM dual UNION ALL
                SELECT to_date('02/01/2017', 'dd/mm/yyyy') date_1 FROM dual UNION ALL
                SELECT to_date('03/01/2017', 'dd/mm/yyyy') date_1 FROM dual UNION ALL
                SELECT to_date('04/01/2017', 'dd/mm/yyyy') date_1 FROM dual UNION ALL
                SELECT to_date('05/01/2017', 'dd/mm/yyyy') date_1 FROM dual)
SELECT d.date_1,
       t1.account_no,
       t1.dept_id,
       t1.currency,
       CASE WHEN t1.amount is NULL THEN
                 LAG(t1.amount, 1, 0) IGNORE NULLS OVER (PARTITION BY t1.account_no, t1.dept_id, t1.currency ORDER BY d.date_1)
            ELSE t1.amount
       END amount,
       t1.date_2
FROM   dates d
       LEFT OUTER JOIN table1 t1 PARTITION BY (t1.account_no, t1.dept_id, t1.currency)
         ON d.date_1 = t1.date_2
ORDER BY d.date_1,
         t1.account_no,
         t1.dept_id,
         t1.currency;

DATE_1      ACCOUNT_NO    DEPT_ID CURRENCY     AMOUNT DATE_2
----------- ---------- ---------- -------- ---------- -----------
01/01/2017           1          1 GBP              45 01/01/2017
01/01/2017           1          1 USD               0 
01/01/2017           1          2 EUR              25 01/01/2017
01/01/2017           1          2 USD               0 
01/01/2017           1          3 USD              51 01/01/2017
02/01/2017           1          1 GBP              45 
02/01/2017           1          1 USD               0 
02/01/2017           1          2 EUR              25 
02/01/2017           1          2 USD              65 02/01/2017
02/01/2017           1          3 USD              51 
03/01/2017           1          1 GBP              45 
03/01/2017           1          1 USD              50 03/01/2017
03/01/2017           1          2 EUR              25 
03/01/2017           1          2 USD              65 
03/01/2017           1          3 USD              51 
04/01/2017           1          1 GBP              45 
04/01/2017           1          1 USD              50 
04/01/2017           1          2 EUR              25 
04/01/2017           1          2 USD              65 
04/01/2017           1          3 USD              51 
05/01/2017           1          1 GBP              45 
05/01/2017           1          1 USD              50 
05/01/2017           1          2 EUR              25 
05/01/2017           1          2 USD              65 
05/01/2017           1          3 USD              51 

N.B。如果您使用的是 11.2 之前的 Oracle 版本,lag 将不知道忽略空值。您可以使用以下代替来模拟相同的效果:

nvl(last_value(t1.amount) IGNORE NULLS OVER (PARTITION BY t1.account_no, t1.dept_id, t1.currency ORDER BY d.date_1), 0)
WITH table1 AS
    (
        SELECT 1 account_no,
            1 dept_id,
            'USD' currency,
            50 amount,
            to_date('03/01/2017', 'dd/mm/yyyy') date_2
        FROM dual
        UNION ALL
        SELECT 1 account_no,
            2 dept_id,
            'EUR' currency,
            25 amount,
            to_date('01/01/2017', 'dd/mm/yyyy') date_2
        FROM dual
        UNION ALL
        SELECT 1 account_no,
            3 dept_id,
            'USD' currency,
            51 amount,
            to_date('01/01/2017', 'dd/mm/yyyy') date_2
        FROM dual
        UNION ALL
        SELECT 1 account_no,
            1 dept_id,
            'GBP' currency,
            45 amount,
            to_date('01/01/2017', 'dd/mm/yyyy') date_2
        FROM dual
        UNION ALL
        SELECT 1 account_no,
            2 dept_id,
            'USD' currency,
            65 amount,
            to_date('02/01/2017', 'dd/mm/yyyy') date_2
        FROM dual
    )
    ,
    dates AS
    (
        SELECT to_date('01/01/2017', 'dd/mm/yyyy') date_1 FROM dual
        UNION ALL
        SELECT to_date('02/01/2017', 'dd/mm/yyyy') date_1 FROM dual
        UNION ALL
        SELECT to_date('03/01/2017', 'dd/mm/yyyy') date_1 FROM dual
        UNION ALL
        SELECT to_date('04/01/2017', 'dd/mm/yyyy') date_1 FROM dual
        UNION ALL
        SELECT to_date('05/01/2017', 'dd/mm/yyyy') date_1 FROM dual
    )
SELECT d.date_1,
    t1.account_no,
    t1.dept_id,
    t1.currency,
    t1.amount,
    CASE (t1.date_2)
        WHEN d.date_1
        THEN t1.date_2
        ELSE NULL
    END
FROM table1 t1,
    dates d
ORDER BY d.date_1,
    t1.account_no,
    t1.dept_id,
    t1.currency;