LEFT OUTER JOIN不显示JOIN右侧缺少的行

时间:2012-04-17 10:52:51

标签: sql oracle11g left-join

我正在构建一个查询以返回给定的员工,该人员角色需要的所有“测试”,此人未参加测试或测试已过期。

第一部分列出了员工角色的所有测试,如下所示:

SELECT e.EMP_NO, e.POS_CODE, p.PROPERTY_CODE, p.PROPERTY_VALUE
FROM COMPANY_PERS_ASSIGN e
INNER JOIN COMPANY_POS_PROPERTY_ALL p ON p.POS_CODE = e.POS_CODE
WHERE e.EMP_NO = '3615';

到目前为止一直很好......对于这个例子,它列出了这个人角色所需的4个测试。

接下来,我需要将其加入到显示员工采取了哪些测试的视图中:

SELECT e.EMP_NO, e.POS_CODE, p.PROPERTY_CODE, p.PROPERTY_VALUE, ep.PROPERTY_CODE, ep.PROPERTY_VALUE
FROM COMPANY_PERS_ASSIGN e
INNER JOIN COMPANY_POS_PROPERTY_ALL p ON p.POS_CODE = e.POS_CODE
LEFT OUTER JOIN COMPANY_EMPLOYEE_PROPERTY_ALL ep ON ep.PROPERTY_VALUE = p.PROPERTY_VALUE
WHERE e.EMP_NO = '3615';

使用LEFT OUTER JOIN,我希望它返回上一个查询中的所有行,以及我们刚添加的视图中的任何匹配信息。

如果此角色的测试是A / B / C / D,并且该员工已经完成了一次测试A,并且测试了D 3次,那么我预计会有以下输出:

3615 POS1 A A A A
3615 POS1 B B
3615 POS1 C C
3615 POS1 D D D D
3615 POS1 D D D D
3615 POS1 D D D D

但我得到了:

3615 POS1 A A A A
3615 POS1 D D D D
3615 POS1 D D D D
3615 POS1 D D D D

我比PL / SQL更熟悉T-SQL,但据我所知,我应该采取正确的方法,但显然有些事情是不对的。

我错过了什么?

更新 我在逻辑中意识到错误 - 在我的原始代码示例中,我加入了COMPANY_EMPLOYEE_PROPERTY_ALL表,通过PROPERTY_VALUE(即测试)进行链接,但没有对该员工进行任何说明。

为了解决这个问题(并尝试不同的东西),我介绍了一个子查询:

SELECT e.EMP_NO, e.POS_CODE, p.PROPERTY_VALUE, p.value_description, ep.PROPERTY_VALUE, ep.value_description, ep.VALID_TO
FROM COMPANY_PERS_ASSIGN e
INNER JOIN COMPANY_POS_PROPERTY_ALL p ON p.POS_CODE = e.POS_CODE
LEFT OUTER JOIN 
    (Select ep.PROPERTY_VALUE, ep.VALID_TO, ep.value_description
    FROM COMPANY_PERS_ASSIGN e
    INNER JOIN COMPANY_EMPLOYEE_PROPERTY_ALL ep ON ep.EMP_NO = e.EMP_NO
    WHERE e.EMP_NO = '3615') ep on ep.PROPERTY_VALUE = p.PROPERTY_VALUE
WHERE e.EMP_NO = '3615'
AND (to_char(ep.VALID_TO) <= '17-APR-12' OR ep.VALID_To IS NULL);

此解决方案有效。

然而,考虑到我发现的问题,我怎样才能重新开始原始方法?或者我应该坚持使用这个解决方案。我正处于“工作”阶段,但我一直在寻找更优雅(并有机会学习)。

1 个答案:

答案 0 :(得分:3)

尝试在左外连接条件中添加与EMP_NO相关的条件:

SELECT e.EMP_NO, e.POS_CODE, p.PROPERTY_CODE, p.PROPERTY_VALUE, ep.PROPERTY_CODE, ep.PROPERTY_VALUE
FROM COMPANY_PERS_ASSIGN e
INNER JOIN COMPANY_POS_PROPERTY_ALL p ON p.POS_CODE = e.POS_CODE
LEFT OUTER JOIN COMPANY_EMPLOYEE_PROPERTY_ALL ep ON ep.PROPERTY_VALUE = p.PROPERTY_VALUE
                                                AND ep.EMP_NO = e.EMP_NO
WHERE e.EMP_NO = '3615'
-- AND (ep.VALID_TO <= to_date('20120417', 'yyyymmdd') OR ep.VALID_To IS NULL)