从 SQL 到 SPL:组内查找最近的匹配记录

MSSQL 数据库的表 mytable,每个 ID 有一个 ConfirmationStarted 和多个 Closed 状态。

|-------------------------|------|---------------------|
| CreatedAt | ID | NewStatus |
| 2022-05-25 23:17:44.000 | 147 | Active |
| 2022-05-28 05:59:02.000 | 147 | Closed |
| 2022-05-30 20:48:53.000 | 147 | Active |
| 2022-06-18 05:59:01.000 | 147 | Closed |
| 2022-06-21 20:09:48.000 | 147 | Active |
| 2022-06-25 05:59:01.000 | 147 | Closed |
| 2022-07-13 00:02:47.000 | 147 | ConfirmationStarted |
| 2022-07-15 15:33:30.000 | 147 | ConfirmationDone |
| 2022-08-25 05:59:01.000 | 147 | Closed |
| 2023-03-08 13:34:57.000 | 1645 | Draft |
| 2023-03-22 19:58:51.000 | 1645 | Active |
| 2023-04-29 05:59:02.000 | 1645 | Closed |
| 2023-05-08 14:50:29.000 | 1645 | Awarded |
| 2023-05-08 14:53:34.000 | 1645 | ConfirmationStarted |
| 2023-05-08 17:53:55.000 | 1645 | ConfirmationDone |

现在要在每个 ID 里,找到 ConfirmationStarted 之前的所有的 Closed 中,离 ConfirmationStarted 最近的那条记录,取出记录的 ID 和时间字段。

|------|-------------------------|
| ID | xdate |
| 147 | 2022-06-25 05:59:01.000 |
| 1645 | 2023-04-29 05:59:02.000 |

SQL 解法:

复制代码
With cte AS (
    SELECT ID, CreatedAt, NewStatus,
        ROW_NUMBER() OVER (PARTITION BY ID ORDER BY CreatedAt DESC) AS rn
    FROM mytable
    WHERE NewStatus = 'Closed'
    AND CreatedAt < (
        SELECT CreatedAt FROM mytable AS sub
        WHERE sub.ID = mytable.ID AND sub.NewStatus = 'ConfirmationStarted'
    )
)
SELECT ID, CreatedAt as xdate
FROM cte
WHERE rn = 1
ORDER BY ID;

SQL没有天然序号,需要先用窗口函数生成序号。SQL分组后必须立刻汇总,不能对组内记录进行过滤,只能绕道用多层子查询反复过滤。整体代码有点繁琐又难懂。

SPL有天然序号,还提供有丰富的与位置相关的计算。SPL分组后可以保持分组子集,便于处理组内数据。

|---|------------------------------------------------------------------------------------|
| |  A |
| 1 | "select ID,CreatedAt,NewStatus from mytable order by CreatedAt")=mssql.query( |
| 2 | =A1.group(ID) |
| 3 | =A2.(~.select@c(NewStatus!="ConfirmationStarted").select@z1(NewStatus=="Closed")) |
| 4 | =A3.new(ID,CreatedAt:xdate) |

A1:从数据库加载数据,按时间排序。

A2:按 ID 分组,但不汇总。

A3:过滤每组数据,先找到 ConfirmationStarted 之前的记录,再从中过滤出 Closed,取倒数第 1 条。函数 select 用于条件过滤,过滤时支持与位置相关的计算,@c 表示从第一个使条件为真的记录开始取,直到遇到使条件为假的记录时停止,@1 表示取结果的第 1 条,@z 表示从后往前过滤。

A2-A4 可以合成一句:=A1.group(ID;~.select@c(NewStatus!="ConfirmationStarted").select@z1(NewStatus=="Closed").CreatedAt:xdate)

SPL已开源免费,欢迎前往乾学院了解更多!

免费下载

相关推荐
小光学长25 分钟前
基于ssm的膳食健康管理系统e6whl4q7(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。
java·开发语言·数据库·学习·ssm
一个天蝎座 白勺 程序猿32 分钟前
KingbaseES融合数据库:一库多能,企业数据管理新思路
数据库·性能优化·kingbasees·金仓数据库
橘颂TA42 分钟前
【MySQL】解锁表的 N 种牵手方式:SQL 连接与子查询漫游(复合查询)
数据库·mysql
数据知道1 小时前
MongoDB基于角色的访问控制(RBAC):精细化权限管理的实用方法
数据库·mongodb
代码派1 小时前
NineData社区版:免费+本地化部署,满足数据库DevOps、数据复制与一致性对比的数据库管理平台
运维·数据库·database·devops·数据库管理工具·ninedata·数据库迁移
jarvisuni1 小时前
GLM5实战测试,挑战Opus4.6 !
前端·数据库
wyt5314291 小时前
基于人脸识别和 MySQL 的考勤管理系统实现
数据库·mysql
数据知道1 小时前
MongoDB分片键选择策略:决定数据分布与查询性能的关键因素
数据库·mongodb
smchaopiao2 小时前
Python数据库操作:SQLAlchemy ORM指南
jvm·数据库·python
baivfhpwxf20232 小时前
ACS X轴回零程序 项目实战版
网络·数据库·算法