SQL 分组条件深入剖析

本文深入剖析一个SQL分组条件问题:如何查找只有Ready状态的设备。通过两种SQL实现方法,揭示SQL在处理此类问题时的不足,探讨更直观的解决方案,并引入集算器SPL语言的简洁表达方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题

在 stackoverflow 网站上看到这样一个 SQL 分组条件的需求,需求看似挺简单,但能把 SQL 写正确对于新手来说也不容易,我们拿过来深入剖析一下,数据如下:

需求是查找只有Ready 状态的设备。

解答

自然思路:按设备分组,选出Ready状态个数等于所有状态个数的设备组

拆解思路:

1、 按设备分组

2、 计算出每个设备组下Ready状态个数c1

3、 计算出每个设备组下所有状态个数c2

4、 选出状态个数c1等于c2的设备;

思路图:

试写SQL一:

select t1.device

from

(select device, count(status) as c1

from t

where status='Ready'

group by device) t1

join

(select device, count(status) as c2

from t

group by device) t2

on t1.device=t2.device

where t1.c1=t2.c2

这个实际的思路变成了:

1、 按设备分组,找到每个设备下Ready的个数c1;

2、 按设备分组,找到每个设备下所有状态的个数c2;

3、 把两个分组结果通过设备连接到一起;

4、 在连接到一起结果里,查找c1等于c2的设备。

试写SQL二:

select device

from t

group by device

having sum(case when status = 'Ready' then 1 else 0 end) = count(device);

这个实际思路,倒是和期望的自然思维更近了些,但计算设备中Ready状态个数c1时,没有找到太简单的表述方式,采用了一个变通的复杂思路:

1、隐含着增加一个计算字段,状态为Ready时,这个计算字段为1,否则为0;

2、然后对这个计算字段求和,这个和是Ready状态的个数。

上面我试了两种SQL,虽然结果都对,但竟然难以找到和自然思路一致的写法。面对这种局面,我们不能只是满足于SQL老手对新手的炫技;要更深入的思考SQL语言体系到底缺失了什么根本概念?才导致自然而然的解题思路,能说的出,想得明白,但就是写不出相应的计算机语言。

先直接看一下集算器SPL语言的解法:

A

1

=connect("mysqlDB")

2

=A1.query(“select * from t”)

3

=A2.group(device)

4

=A3.select(~.select(status=="Ready").len()==~.len())

5

=A1.close()

A2从数据库把数据取出来

A3按设备分组,观察一下结果,就只是把数据分成了5组,第一组点进去,能看到有3条符合条件的记录,这一步仅仅做单纯分组,没有其它任何操作:

A4外层select函数里的~符号代表的就是当前设备的组;~.select(status=="Ready").len()就是当前设备下Ready状态个数c1;~.len()是当前设备所有状态的个数 c2;这时最外层 select 变成 A3.select(c1==c2), 也就是自然思路里的第四步:选出状态个数c1等于c2的设备。

完全符合人的自然思维!编码过程顺畅自然,两点之间,直线最短,不用再通过各种“高级技巧”的等价思路绕弯了。

追根溯源,上面 SQL 反映出来的问题,它缺少单纯的分组动作,不能显式的表达每个小分组,进而对每个小分组的更细致操作就无法直观表达。

SQL不提倡分步解决问题、对集合操作支持不彻底、多余的生造出 HAVING 人为的导致概念复杂化;这一系列设计上的缺陷,造成了 SQL 的编写、维护、性能调优都成倍的增加工作量。

集算器的 SPL 语言从根源上弃用了 SQL 背后的关系代数理论,发明新的离散数据集理论解决程序员描述计算的困难。这里有更多 SQL 难点的分析文章:

乾学院 -SPL-SQL 难点

快速上手试试:

下载集算器

如何免费使用润乾集算器

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值