4

我有一张表格,其中列出了员工活跃/不活跃的日期,我想计算员工在特定日期范围内活跃的周数。

因此表 (ps_job) 将具有如下值:

EMPLID     EFFDT       HR_STATUS
------     -----       ------
1000       01-Jul-11   A
1000       01-Sep-11   I
1000       01-Jan-12   A
1000       01-Mar-12   I
1000       01-Sep-12   A

查询需要向我显示此 empid 从 2011 年 7 月 1 日到 2012 年 12 月 31 日活跃的周数。

所需的结果集将是:

EMPLID     WEEKS_ACTIVE
------     ------------
1000       35

通过添加以下 SQL 的结果,我得到了数字 35:

SELECT (NEXT_DAY('01-Sep-11','SUNDAY') - NEXT_DAY('01-Jul-11','SUNDAY'))/7 WEEKS_ACTIVE FROM DUAL;
SELECT (NEXT_DAY('01-Mar-12','SUNDAY') - NEXT_DAY('01-Jan-12','SUNDAY'))/7 WEEKS_ACTIVE FROM DUAL;
SELECT (NEXT_DAY('31-Dec-12','SUNDAY') - NEXT_DAY('01-Sep-12','SUNDAY'))/7 WEEKS_ACTIVE FROM DUAL;

问题是我似乎无法弄清楚如何创建一个查询语句,该语句将遍历特定日期范围内每个员工的所有行,并只返回每个 empid 以及它们处于活动状态的周数。我更喜欢使用基本 SQL 而不是 PL/SQL,以便我可以将其传输到可由用户运行的 PeopleSoft 查询,但如果需要,我愿意使用 Oracle SQL Developer 为用户运行它。

数据库:Oracle Database 11g Enterprise Edition Release 11.2.0.3.0 - 64bit Production

4

3 回答 3

0

如果客户只是想要一个粗略的估计,我会从每次工作的天数开始,除以 7 并四舍五入。

诀窍是将活动日期与其相应的非活动日期对齐,我能想到的最好方法是分别挑选活动日期和非活动日期,按日期对它们进行排名,然后将它们重新组合在一起EmplID并进行排名。在这种ROW_NUMBER()情况下,分析函数是最好的排名方式:

WITH
  EmpActive AS (
    SELECT
        EmplID,
        EffDt,
        ROW_NUMBER() OVER (PARTITION BY EmplID ORDER BY EffDt NULLS LAST) DtRank
      FROM ps_job
      WHERE HR_Status = 'A'
  ),
  EmpInactive AS (
   SELECT
      EmplID,
      EffDt,
      ROW_NUMBER() OVER (PARTITION BY EmplID ORDER BY EffDt NULLS LAST) DtRank
    FROM ps_job
    WHERE HR_Status = 'I'
  )
SELECT
  EmpActive.EmplID,
  EmpActive.EffDt AS ActiveDate,
  EmpInactive.EffDt AS InactiveDate,
  ROUND((NVL(EmpInactive.EffDt, TRUNC(SYSDATE)) - EmpActive.EffDt) / 7) AS WeeksActive
FROM EmpActive
LEFT JOIN EmpInactive ON
    EmpActive.EmplID = EmpInactive.EmplID AND
    EmpActive.DtRank = EmpInactive.DtRank

第三次演出EmplID = 1000有一个活动日期但没有非活动日期,因此两个子查询之间NULLS LASTROW_NUMBER排序和左连接。

我在这里使用了“days / 7”数学;当您收到客户的回复时,您可以替换您需要的东西。请注意,如果没有相应的非活动日期,则查询使用当前日期。

这里有一个 SQLFiddle 。

于 2013-05-08T17:58:54.007 回答
0

以下内容应该适用于您正在尝试做的事情。我确实必须在 NVL 声明中硬编码结束日期

SELECT emplid,
       hr_status,
       ROUND(SUM(end_date - start_date)/7) num_weeks
  FROM (SELECT emplid,
               hr_status,
               effdt start_date,
               NVL(LEAD(effdt) OVER (PARTITION BY emplid ORDER BY effdt), 
                                        TO_DATE('12312012','MMDDYYYY')) end_date
          FROM ps_job
       )
 WHERE hr_status = 'A'
 GROUP BY emplid,
          hr_status
 ORDER BY emplid

内部查询将从表中提取员工和 HR 状态信息,并使用 effdt 列作为开始日期,并使用 LEAD 分析函数从表中获取下一个 effdt 日期值,它表示下一个状态的开始等将是当前行的 end_date。如果 LEAD 函数返回 NULL,我们会为其分配您想要的完成日期 (12/31/2012)。然后,他的 out 语句仅将结果集限制为具有活动 HR 状态的记录并计算周数。

于 2013-05-08T19:01:30.877 回答
0

在这里,我lead在子查询中使用来获取下一个日期,然后对外部查询中的间隔求和:

with q as (
    select EMPLID, EFFDT, HR_STATUS
        , lead (EFFDT, 1) over (partition by EMPLID order by EFFDT) as NEXT_EFFDT
    from ps_job
    order by EMPLID, EFFDT
)
select EMPLID
    , trunc(sum((trunc(coalesce(NEXT_EFFDT, current_timestamp)) - trunc(EFFDT)) / 7)) as WEEKS_ACTIVE
from q
where HR_STATUS = 'A'
group by EMPLID;

coalesce如果找不到匹配的I记录(员工是当前员工) ,该功能将获取系统日期。如果那是您的规格,您可以替换年底。

请注意,我没有进行任何严格的测试来查看您的条目是否按 A/I/A/I 等排序,因此如果您知道您的数据需要它,您可能需要添加这种性质的检查。

随意在SQL Fiddle上玩这个。

于 2013-05-08T19:36:52.433 回答