mysql 迪卡尔运算_Mysql经典练习题拆解思路(二) _____笛卡尔积

博客通过分析力扣上的"分数排名"题目,深入讲解了如何利用MySQL处理笛卡尔积问题。通过示例解释了当进行表连接时如何避免产生笛卡尔积,并展示了GROUP BY和DISTINCT在解决此类问题中的作用,最终实现按排名正序排列数据。

开发板推荐:天空星STM32F407VET6开发板

超高性价比 STM32主控 | 超高主频 | 一板兼容百芯 | 比赛神器 | 沉金彩色丝印

年底不忙, 还没放假,闲来无事刷数据库题玩玩.看到一个题,做完了觉得很有收获.也加深了对笛卡尔积的理解. 现记录在此

题目是力扣的, 叫做"分数排名"

b63cded72575b779ae3b98ff8e5acd76.png

网上一个大牛的写法:

select s1.Score,count(distinct(s2.Score)) as Rank from

Scores s1,Scores s2

where

s1.Score <= s2.Score

group by s1.Id

order by Rank

数据库表连接数据行匹配时就会产生笛卡尔积,关于笛卡尔积的定义,引自百度百科:

笛卡尔乘积是指在数学中,两个集合_X_和_Y_的笛卡尔积(Cartesian product),又称直积,表示为_X_×_Y_,第一个对象是_X_的成员而第二个对象是_Y_的所有可能有序对的其中一个成员[3]

8508d7797ca035b4b9e62a9a7d55388d.png

如果在数据库中进行表连接就会产生一个笛卡尔积临时表,在这个例子中如下:

a42e9e527dd7a05a182591e935cc633a.png

全部记录应该是36行*4列

查询第一条记录时: Id: 1 ,Score 3.50 当限定where s1.Score <= s2.Score时,结果集为:

{(3.50,3.50),(3.50,3.65),(3.50,4.00),(3.50,3.85),(3.50,4.00),(3.50,3.65)}

GROUP BY Id为1,即分出一条记录存在Id为1的数据集, 如果不加GROUP BY Id, 则会取所有结果集中最小的一条记录,而不是分组展示.

最后用distinct去重s2分数 剩下4个,取count,即 Rank为4

{(3.50,3.50),(3.50,3.65),(3.50,4.00),(3.50,3.85)}

第二条纪录: Id:2, Score: 3.65 当where s1.Score <= s2.Score时

{(3.65,3.65),(3.65,4.00),(3.65,3.85),(3.65,4.00),(3.65,3.65)}

{(3.65,3.65),(3.65,4.00),(3.65,3.85),(3.65,3.85),(3.65,3.65)}

GROUP BY Id为2, 分出一条记录存在Id为2的数据集

然后去重s2分数 剩下3个, 即Rank为3

{(3.65,3.65),(3.65,4.00)}

{(3.65,3.85)}

以此类推..

最后 order by Rank 按排名正序排列数据

如果把order by Rank去掉,得到下面结果集:

[[3.50, 4], [3.65, 3], [4.00, 1], [3.85, 2], [4.00, 1], [3.65, 3]]

每条数据内的分值默认是按照表的Id排列, 由此证明 上面分析的查询顺序没错

开发板推荐:天空星STM32F407VET6开发板

超高性价比 STM32主控 | 超高主频 | 一板兼容百芯 | 比赛神器 | 沉金彩色丝印

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值