oracle中的group by和union

Posted

技术标签:

【中文标题】oracle中的group by和union【英文标题】:group by and union in oracle 【发布时间】:2015-06-08 14:46:45 【问题描述】:

我想合并 2 个查询,但在 oracle 中遇到错误。

select count(*) as faultCount,
       COMP_IDENTIFIER 
from CORDYS_NCB_LOG 
where AUDIT_CONTEXT='FAULT' 
union 
select count(*) as responseCount,
       COMP_IDENTIFIER 
from CORDYS_NCB_LOG 
where AUDIT_CONTEXT='RESPONSE' 
group by COMP_IDENTIFIER  
order by responseCount; 

两个查询单独运行完美。但是在使用联合时,它显示 ORA-00904: "RESPONSECOUNT": invalid identifier

【问题讨论】:

您正在创建一个带有计数列的结果集。您正在为同一列定义 2 个名称。有一个正确答案,我只是想知道你是要单列还是两列。 我需要 2 列 faultCount,responseCount 【参考方案1】:

你遇到的错误

在 Oracle 中,最好始终以相同的方式命名每个 UNION 子查询中的每一列。在您的情况下,以下应该有效:

select count(*) as theCount,
       COMP_IDENTIFIER 
from CORDYS_NCB_LOG 
where AUDIT_CONTEXT='FAULT' 
group by COMP_IDENTIFIER -- don't forget this
union 
select count(*) as theCount,
       COMP_IDENTIFIER 
from CORDYS_NCB_LOG 
where AUDIT_CONTEXT='RESPONSE' 
group by COMP_IDENTIFIER  
order by theCount; 

另见:

Curious issue with Oracle UNION and ORDER BY

当然,一个好的解决方法是使用 a_horse_with_no_name 建议的索引列引用

你真正想要的查询

但是,根据您的 cmets,我怀疑您想编写一个完全不同的查询,即:

select count(case AUDIT_CONTEXT when 'FAULT'    then 1 end) as faultCount,
       count(case AUDIT_CONTEXT when 'RESPONSE' then 1 end) as responseCount,
       COMP_IDENTIFIER 
from CORDYS_NCB_LOG 
where AUDIT_CONTEXT in ('FAULT', 'RESPONSE')
group by COMP_IDENTIFIER  
order by responseCount; 

【讨论】:

上面写着 ORA-00937: not a single-group group function 谢谢。它可以工作。但是有没有办法增强它。检索需要10多秒,以防万一是不可接受的。 @user1650864:你应该有一个索引:CREATE INDEX my_index ON CORDYS_NCB_LOG(COMP_IDENTIFIER, AUDIT_CONTEXT) 仍然需要 16 秒! 哦,也许,一个额外的 AUDIT_CONTEXT in ('FAULT', 'RESPONSE') 谓词可能会起作用......【参考方案2】:

联合的列名由first查询决定。所以你的第一列实际上被命名为FAULTCOUNT

但是对联合结果进行排序最简单的方法是使用列索引:

select ...
union 
select ...
order by 1;

您很可能还想使用UNION ALL,它可以避免删除两个查询之间的重复项,并且比普通的UNION 更快

【讨论】:

“联合的列名由第一个查询确定” - 事实上,这并不完全正确(对于 Oracle)。很可能ORDER BY 中没有可用的列名。 This might just be a bug, though 我需要结果集列是 faultCount,responseCount,COMP_IDENTIFIER 那么你不需要一个 UNION。 UNION 说构建一个数据集,将来自第二个查询的数据附加到来自第一个查询的数据集的末尾 - 并将列对齐。 JOIN 用于将两个查询放在同一行上。 @user1650864 union 只有两列。 one 列只能有 one 名称。您不能同时命名第一列 FAULTCOUNT COMP_IDENTIFIER 嗨,你能发布你试图解释的查询吗【参考方案3】:

在 Union 或 Union 中,所有查询列名都由第一个查询列名确定。

在您的查询中,将“order by responseCount”替换为“order by faultCount”。

【讨论】:

以上是关于oracle中的group by和union的主要内容,如果未能解决你的问题,请参考以下文章

oracle group by 性能优化

oracle sql中的group by表达式的内部连接[重复]

[Database] Oracle 中的where 可以后接group by

Oracle Group by NULL 中的 SQL 返回多行

oracle语句中的聚合函数以及分组group by的使用实例

如何在不使用 GROUP BY 或 PARTITION BY 的情况下对 Oracle SQL 中的数据进行分组