[英]How do I perform the following SQL (or PL/SQL) Query
如果面临以下要求:用户需要一个表,在该表中他可以看到员工 ID、员工姓名和员工不工作的天数。 原始数据存储在下表中:
Employee: Emp_ID, Emp_Name
WorkDays: Emp_ID, Date
该过程是每当员工工作时,他的 ID 和当天的日期都会插入 WorkDays 表中。
现在我需要一个 SQL 或 PL/SQL 查询,在其中我选择一个日期范围(fe 介于 2021 年 10 月 1 日和 2021 年 11 月 1 日之间)以及在此期间至少有 1 天没有工作的每个员工(意味着有WorkDays 表中“缺少”特定 ID 和日期的一行),他将显示在一行中,其中包含他没有工作的日期。 日期都在一行的一列中,fe 用逗号分隔。
为了更好地理解,如果有两名员工在那段时间没有工作,那么决赛桌应该是这样的:
Emp_ID | Emp_Name | Hasn't Worked |
1 | John | 2021-10-01, 2021-10-03, 2021-10-06 |
3 | Mary | 2021-10-02, 2021-10-03 |
我不希望我的代码为我完成,而是在查询表的问题上获得帮助,在该表中我必须循环遍历数据并在相应的行中显示多个结果(日期)。
非常感谢您的帮助! 杰克
使用行生成器创建日历,然后使用PARTITION OUTER JOIN
并找到不匹配的行,然后聚合:
WITH calendar (day) AS (
SELECT DATE '2021-10-01' FROM DUAL
UNION ALL
SELECT day + 1 FROM calendar WHERE day < DATE '2021-11-01'
)
SELECT w.emp_id,
MAX(emp_name) AS emp_name,
LISTAGG(TO_CHAR(c.day, 'YYYY-MM-DD'), ', ') WITHIN GROUP (ORDER BY c.day)
AS non_work_days
FROM calendar c
LEFT OUTER JOIN workdays w
PARTITION BY (w.emp_id)
ON (c.day = w."DATE")
LEFT OUTER JOIN employee e
ON (w.emp_id = e.emp_id)
WHERE w."DATE" IS NULL
GROUP BY w.emp_id;
或者您可以使用CROSS JOIN
:
WITH calendar (day) AS (
SELECT DATE '2021-10-01' FROM DUAL
UNION ALL
SELECT day + 1 FROM calendar WHERE day < DATE '2021-11-01'
)
SELECT e.emp_id,
MAX(e.emp_name) AS emp_name,
LISTAGG(TO_CHAR(c.day, 'YYYY-MM-DD'), ', ') WITHIN GROUP (ORDER BY c.day)
AS non_work_days
FROM calendar c
CROSS JOIN employee e
LEFT OUTER JOIN workdays w
ON (c.day = w."DATE" AND e.emp_id = w.emp_id)
WHERE w."DATE" IS NULL
GROUP BY e.emp_id;
其中,对于样本数据:
CREATE TABLE Employee (Emp_ID, Emp_Name) AS
SELECT 1, 'Alice' FROM DUAL UNION ALL
SELECT 2, 'Beryl' FROM DUAL;
CREATE TABLE WorkDays (Emp_ID, "DATE") AS
WITH calendar (dt) AS (
SELECT DATE '2021-10-01' + LEVEL - 1 FROM DUAL CONNECT BY LEVEL <= 32
)
SELECT 1, dt FROM calendar WHERE dt NOT IN (DATE '2021-10-01', DATE '2021-10-03', DATE '2021-10-06')
UNION ALL
SELECT 2, dt FROM calendar WHERE dt NOT IN (DATE '2021-10-02', DATE '2021-10-03')
output:
EMP_ID EMP_NAME NON_WORK_DAYS 1 爱丽丝 2021-10-01, 2021-10-03, 2021-10-06 2 绿柱石 2021-10-02, 2021-10-03
db<> 在这里摆弄
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.