首页 > 代码库 > SQL学习之HAVING过滤分组

SQL学习之HAVING过滤分组

1、SQL除了能用Group By分组数据之外,SQL还允许过滤分组,规定包括那些分组,排除那些分组。例如,你可能想要列出至少有两个订单的所有顾客。为此,必须基于完整的分组而不是个别的行进行过滤。

    基于上面的例子,我们第一时间想到的是通过使用WHERE来过滤数据,拿到我们想要的结果,但是在这个列子中WHERE不能完成任务,因为WHERE过滤指定的是行,而不是分组。事实上,WHERE没有分组的概念。

    因此,SQL提供了专门用来过滤分组的类似与WHERE子句的子句,HAVING子句,事实上,几乎所有类型的WHERE子句都可以用HAVING来代替。唯一的差别是,WHERE过滤行数据,HAVING过滤分组数据。

如下代码:

select * from dbo.T_Unit_Equipment

技术分享

现在有一个报表系统,需要展示每个单位,拥有每台设备的数量,这个时候用WHERE很麻烦(前提是数据量比较少的情况下),下面是解决代码:

select UnitId,EquipmentName,COUNT(*) as Equipments from dbo.T_Unit_Equipment group by UnitId,EquipmentName order by UnitId ASC

技术分享

ok,完成需求!

现在,改变需求,现在需要检索出每家单位拥有的同一种设备大于等于2的记录,下面是解决代码:

select UnitId,EquipmentName,COUNT(*) as Equipments from dbo.T_Unit_Equipment group by UnitId,EquipmentName HAVING COUNT(*)>=2 order by UnitId ASC 

技术分享

ok,完成需求!

上面的代码通过UnitId,EquipmentName两个字段进行分组,然后SELECT语句检索出UnitId,EquipmentName,以及Equipments(Equipments在这里是一个计算字段,通过COUNT()函数计算出每个分组的总记录数),然后HAVING子句告诉SELECT语句只检索出所有分组中的总记录数大于等于2的分组记录。这里的COUNT(*)>=2过滤了每家单位所拥有的同一种设备小于2的分组!

 

2、WHERE和HAVING的差别:

WHERE在数据分组前进行过滤,HAVING在数据分组之后进行过滤,这是个很重要的区别,WHERE排除的行不包括在分组中。这可能会改变计算值,从而影响HAVING子句中基于这些值的过滤掉的分组,根据这个差别,我们可以确定WHERE子句和Group By子句的位置:WHERE子句会在数据分组之前,对行数据进行过滤。下面通过一个例子来了解HAVING子句与WHERE子句的差别

select * from dbo.T_Unit_Equipment

技术分享

现在需要检索出每家单位拥有的同一种设备大于等于2的记录,前提是设备必须有维护人员(对应表中的Person列值不能为空),下面是解决代码:

select UnitId,EquipmentName,COUNT(*) as Equipments from dbo.T_Unit_Equipment where Person!=‘‘ group by UnitId,EquipmentName HAVING COUNT(*)>=1 order by UnitId ASC 

技术分享

SQL学习之HAVING过滤分组