SQL 分组条件深入剖析

article/2025/10/17 6:02:44

问题

在 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 难点

快速上手试试:

下载集算器

如何免费使用润乾集算器


http://chatgpt.dhexx.cn/article/YJsFF0tx.shtml

相关文章

SQL分组排序函数(组内分别排序)

建表并插入数据 -- 部门表 create table dept( deptno int primary key auto_increment, -- 部门编号 dname varchar(14) , -- 部门名字 loc varchar(13) -- 地址 ) ; -- 员工表 create table emp( empno int primary key auto_increment,-- 员工编号 …

sql 分组 行列转换

sql 分组 & 行列转换 文章目录 sql 分组 & 行列转换1、groupby(配合组合函数使用)2、Sql的行列转换 - 纵横表1)纵表转横表2)横表转纵表 sql语句教程参考W3C School - SQL 教程 就够了 1、groupby(配合组合函数…

SQL:分组数据

分组数据: A. SQL Server Group By语句 Group By 从字面意义上理解就是根据“By”指定的规则对数据进行分组,所谓的分组就是将一个“数据集” 划分成若干个“小区域”,然后针对若干个“小区域”进行数据处理。 以下是 GROUP BY 子句的语…

SQL语言中的分组数据

(1)group by子句 group by 根据by指定的规则对数据进行分组。 分组:即将一个“数据集”划分成若干个“小区域”,再对若干个“小区域”进行数据处理。 语法: group by 子句为列中的每个值组合生成一个组。 group by子…

SQL分组指南

目录 什么是SQL分组? SQL GROUP BY和Sum 排序分组结果 HAVING和GROUP BY 包含多个表的GROUP BY 按SUM()排序 带有表达式的GROUP BY SQL GROUP BY与DISTINCT 结论 什么是SQL分组? 在SQL中,分组是唯一的列值组合。当查询具有GROUP BY…

单例模式的使用和应用场景

1.概念 标题单例模式:单例指的是单实例,一个类中有且仅有创建一个实例 单例模式的应用场景:windows的任务管理器(不可打开两次吧)、回收站等 单例模式应用一般发现在以下条件下: servlet单例、struts2多例、springmvc单例 &…

单例模式实战应用

理论 什么是单例模式 保证整个系统中一个类只有一个对象的实例,实现这种功能的方式就叫单例模式 常用的 service 和 dao 层的对象通常都是单例的,而多例则指每个请求用一个新的对象来处理,比如 action spring 中的 bean 和 spring mvc 中…

单例模式php应用场景,php单例模式 使用场景和使用方法

一个类只有一个对象实例 1、含义 作为对象的创建模式,单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统全局地提供这个实例。它不会创建实例副本,而是会向单例类内部存储的实例返回一个引用。 2、单例模式的三个要点&#xff1a…

Java设计模式及应用场景之《单例模式》

文章目录 一、单例模式定义二、单例模式的结构和说明三、懒汉式和饿汉式的实现1、懒汉式2、饿汉式 四、懒汉式和饿汉式的优缺点五、双重检查加锁方式的实现六、类级内部类方式的实现七、枚举方式的实现 (最佳方式)八、单例模式的应用场景 一、单例模式定义 保证一个类只能有一个…

一文带你了解 Java 五种单例模式的实现方式以及应用场景

单例模式 什么是单例模式 类的单例设计模式,就是采取一定的方法保证在整个软件系统中,某个类只能存在一个对象实例,并且这个类会提供一个获取对象实例的方法。 思路:如果让一个类在一个虚拟机里面只能产生一个对象,就…

js设计模式之 单例模式与应用场景

1.介绍 单例模式(Singleton Pattern)是设计模式中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。 这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个…

单例模式的理解?单例模式如何实现?单例模式应用场景

说说你对单例模式的理解?如何实现? 一、是什么 单例模式(Singleton Pattern):创建型模式,提供了一种创建对象的最佳方式,这种模式涉及到一个单一的类,该类负责创建自己的对象&…

设计模式之单例模式应用场景篇

应用场景 我们为什么要使用单例模式呢?它有什么好处? (一)单例模式可以让我们只创建一个对象从而避免了频繁创建对象导致的内存消耗和垃圾回收。 Servlet是单例模式,我们只需要创建一个Servlet,然后接收请求…

关于getText()的小问题

由一个作业开始的,整完广度优先小作业的时候开始是在代码中指定值进行寻找路径,后面想想还是弄两文本框输入起点和终点更灵活一点好了。谁知道这个JTextField真的让我崩溃了 怎么说应该是我对Java的基础知识没有进行深入了解吧,好吧,我是在今天才知道getText()是在监听事件…

js gettext

test.php 1 <?php2 $localezh_CN;3 if(isSet($_GET["locale"]))$locale $_GET["locale"];4 ?>5 <html>6 <head>7 <link rel"gettext" type"application/x-po" href"./locale/<?php echo $locale ?&…

关于Java getText()方法的问题

这是一个登陆界面的鼠标事件&#xff0c;获取文本区t61的内容&#xff0c;与用户姓名比对&#xff0c;同时验证登陆密码&#xff0c;但是用户姓名比对一直不成功&#xff0c;导致鼠标事件不能反映&#xff0c;页面不能跳转&#xff0c;想知道为什么&#xff0c;求大佬指点。 p…

java的gettext_JAVA里 getText() 是什么意思,怎么用?

展开全部 JAVA里 getText() 的意思是&#xff1a;返回数据窗口控件中 悬浮在当前行列之上的32313133353236313431303231363533e4b893e5b19e31333431366266编辑框中的文本。 在用户修改数据窗口中某项的值时&#xff0c;实际上首先在悬浮 在当前项上的编辑框中进行修改&#xff…

python gettext的使用方法

app.py 里面的print函数输出内容&#xff0c;是会更据你选择的语言平台而改变 将app.py生成.po文件(或者生成.pot文件)&#xff0c;如果是.pot文件需要重命名为.po pygettext.py安装python的时候就自己带着有&#xff0c;去你的安装路径找 python &#xff08;这一步&#xff…

getText()和getText().toString()

在deadline只有一根咪咪的距离时&#xff0c;我就直接吐槽了。 一个组件&#xff0c;比方说EditView&#xff0c;它的getText()和getText().toString()这两种想得到这个组件中的内容的方法是不一样的&#xff0c;我在用startActivityForResulr()之后接收到来自另一个activity的…

登陆QQ时总显示QQ安全防护进程,而且点了确定后QQ还是登不上

解决办法&#xff1a;把服务QPcore启动&#xff0c;或者删除掉防护进程&#xff08;QPcore&#xff09;