原标题:愁!MySQL中如何查询中位数?

导读

计算中位数可能是小学的内容,然而在数据库查询中实现却并不是一件容易的事。我们今天就来看看都有哪些方法可以实现。

781c99cb40485bb830fb34dd86c5e226.png

注:本文所用MySQL版本无限制,所列题目均来源于LeetCode。

LeetCode数据库题目中关于中位数的主要有两道题,难度都是hard级别。两道题目无论是出现频率还是相关企业标签数,都属于比较靠前的位置,包括题解和讨论数量也是如此,足以见其热门程度。

569# 员工薪水中位数

题目描述:

8728498bb324d00dd2d0f2e389ffdcd7.png

预期答案:

23fee2cf917dd467244413216dc3170b.png

解法1

既然是求解中位数,我们首先想到的是根据中位数的定义进行求解:奇数个数字时,中位数是中间的数字;偶数个数字时,中位数中间两个数的均值。本题不进行求解均值,而是将两个中位数全部显示。

根据定义,为了查询中位数,我们需要知道3点信息:

总数是奇数个还是偶数个

待查找数字总数

每个数字的排序编号

前两点信息在MySQL中非常简单,只需简单的count计数即可,而排序编号则需要借助辅助方法。在MySQL8.0以上版本引入了窗口函数后非常容易实现,但以前的版本则仅可通过自定义变量的方式获得排序值。这里如何对员工薪水进行分组排序不再展开,具体可参考历史文章一文解决所有MySQL分类排名问题 。

在有了排名和数字总数之后,如何判断是中位数呢?这里计数字总数为N,则

N为奇数,中位数排序编号是(N+1)/2=N/2+0.5

N为偶数,中位数排序编号是N/2和N/2+1

进一步地,N为奇数和N为偶数是互斥的,求解出的中位数排序编号也是互斥的,也就是说3个排序编号不会同时取得整数,从而可以不加区分的直接判断即可。

查询SQL语句:

1SELECT

2e1.Id, e1.Company, e1.Salary

3FROM

4(SELECT Id, Company, Salary, @rnk:=if(@pre=Company, @rnk+1, 1) rnk, @pre:=Company

5FROM Employee, (SELECT @rnk:=0, @pre:=null)init

6ORDER by Company, Salary, Id)e1

7JOIN

8(SELECT Company, count(*) cnt FROM Employee GROUP by Company) e2

9using(Company)

10WHERE e1.rnk in (cnt/2+0.5, cnt/2, cnt/2+1)

查询效率:

7a102e751d7a555ee7b464112e2ecf9d.png

解法2

除了根据中位数的排序编号来定位其位置,实际上还可以换种思路但仍然是在其排序编号上做文章:如果一个数是中位数,那么就意味着正序和逆序时其位置是一致的:更严谨的说,奇数个数字是正逆序排序一致,偶数个数字时,两中位数顺序要互换一下,也就是相差为1。进而,我们发现无论数字总数是奇数还是偶数,中位数的正逆排序相差要么为0,要么为1。根据这一性质,我们分别实现正逆两遍排序,然后判断数字的排序编号即可。

查询SQL语句:

1SELECT

2e1.Id, e1.Company, e1.Salary

3FROM

4(SELECT Id, Company, Salary, @rnk:=if(@pre=Company, @rnk+1, 1) rnk, @pre:=Company

5FROM Employee, (SELECT @rnk:=0, @pre:=null)init

6ORDER by Company, Salary, Id)e1

7JOIN

8(SELECT Id, Company, Salary, @rnk:=if(@pre=Company, @rnk+1, 1) rnk, @pre:=Company

9FROM Employee, (SELECT @rnk:=0, @pre:=null)init

10ORDER by Company, Salary DESC, Id DESC)e2

11on e1.Id=e2.Id

12WHERE abs(e1.rnk - e2.rnk)<=1

查询效率:

1f247e50ba68dab9c8cb26db6f3496be.png

解法3

前2种解法都是根据中位数的定义在数字排序编号上作文章,下面是一个对中位数性质更深的理解(摘抄自官方题解)

根据定义,我们来找一下 [1, 3, 2] 的中位数。首先 1 不是中位数,因为这个数组有三个元素,却有两个元素 (3,2) 大于 1。3 也不是中位数,因为有两个元素小于 3。对于 2 来说,大于 2 和 小于 2 的元素数量是相等的,因此 2 是当前数组的中位数。当数组长度为 偶数,且元素唯一时,中位数等于排序后 中间两个数 的平均值。对这两个数来说,大于当前数的数值个数跟小于当前数的数值个数绝对值之差为 1,恰好等于这个数出现的频率。

结论:不管数组长度是奇是偶,也不管元素是否唯一,中位数出现的频率一定大于等于 大于它的数和 小于它的数的绝对值之差。

好吧,力扣的官方题解读起来总是这么生涩。不过细品之下,我们还是可以发现这个结论是对的。【好像说了句废话】

根据中位数的这一性质,可以写出如下查询语句:

1SELECT

2e1.Id, e1.Company, e1.Salary

3FROM

4Employee e1,

5Employee e2

6WHERE

7e1.Company = e2.Company

8GROUP BY e1.Company , e1.Salary

9HAVING SUM(e1.Salary = e2.Salary) >= ABS(SUM(SIGN(e1.Salary - e2.Salary)))

10ORDER BY e1.Id

查询效率:

f0273b34a777fb410506b6f43abba3df.png

实际上,虽然3种解法均为两表关联,但由于解法3中涉及到相对更为复杂的计算,其效率竟然要比解法1和解法2中低太多。

所以,不妨想想奥卡姆剃刀原理,大道至简、大巧不工、简单之美!

571# 给定数字的频率查询中位数

刚才一道题是对给定的一组数字查询中位数,顶多也就是要进行分组后查询中位数。那如果给定的数字不是数字全样本,而是数字+频率呢?

题目描述:

0f49a2eebc61a8eaeed66febf2538b6a.png

注:与前一题不同,本题中如果中位数有两个,返回的是一个均值。

解法1

这一题乍一看还是挺懵的,但有了第一题解法3中的结论,似乎它就是为这一题做的铺垫:这不刚好就是提供的数字及其频率吗?对比其小的数字频率求和就是比其小的数字个数,类似的也可以得到比其大的数字个数。

这样的想法其实非常适合窗口函数,如果是在8.0以上版本,那么如下SQL语句可谓是简洁优雅:

1SELECT

2number

3FROM

4(SELECT number, Frequency,

5sum( Frequency ) over (rows BETWEEN unbounded preceding AND current ROW ) cnt1,

6sum( Frequency ) over (rows BETWEEN current ROW AND unbounded following ) cnt2

7FROM

8numbers ) tmp

9WHERE

10Frequency >= abs(cnt2 - cnt1)

其中:cnt1为当前行之前的累计个数(含当前行),cnt2为当前行之后的累计个数(含当前行),进而cnt2-cnt1等于比其大的数字和比其小的数字个数之差。

当然,当前LeetCode OJ是5.6版本,MySQL也不能使用窗口函数。此时,可以简单的通过自定义变量得到实现:

1SELECT

2avg(number) median

3FROM

4(SELECT number, Frequency, @cnt:=@cnt+Frequency cnt

5FROM numbers, (SELECT @cnt:=0)init

6ORDER BY number)tmp1,

7(SELECT sum(Frequency) total FROM numbers)tmp2

8WHERE

9Frequency>=abs(total-2*cnt+Frequency)

类似的,这里:

cnt为当前行之前的累计数字个数(含当前行),cnt-Frequency为不含当前行的数字个数

total为总的数字个数,total-cnt即为当前行之后的数字个数(不含当前行)

total-cnt - cnt+Frequency即为需要求的差值

查询效率:

472c64c3c3b2e8662769e1c29da89108.png

解法2

前面的方法是借助了中位数的一个性质,实话说还是不够直观。那么,如果仍然沿用中位数排序编号的规律,是否可以用于本题的SQL查询呢?

当然可以。实际上,根据数字及频率,可以稍微变形得到数字排序编号的首末区间,然后判断中位数的编号存在于哪个数字的首末区间即可找到中位数。

带着这一想法,我们首先写出如下SQL语句来获得数字的首末区间:

1SELECT

2number, frequency, @beg:=@end+1 AS beg, @end:=@beg+frequency-1 AS end

3FROM

4numbers, (SELECT @beg := 0, @end :=0) init

5ORDER BY

6number

得到如下中间结果:

c72f91a240f116f817142960e8d71eda.png

然后,对中位数位置的三个可能取值(即N/2, N/2+0.5, N/2+1)分别判断是否存在首末区间,进而判断是否是中位数:

1SELECT

2avg(number) median

3FROM

4( SELECT

5number, frequency, @beg := @end+1 AS beg, @end := @beg+frequency-1 AS end

6FROM

7numbers, (SELECT @beg := 0, @end :=0) init

8ORDER BY

9number

10) t1,

11( SELECT sum(frequency) cnt FROM numbers ) t2

12WHERE

13(cnt/2 BETWEEN beg AND end)

14or (cnt/2+0.5 BETWEEN beg AND end)

15or (cnt/2+1 BETWEEN beg AND end)

查询效率:

e2f828d83bdf2b6b27ec46caea35e427.png

解法3

利用中位数的排序值可以判断,利用正逆序的差值应该也可以。仍然是通过正逆两遍排序得到每个数字的两组首末区间,然后 判断两个区间在相差1范围内是否存在交集即可。

查询SQL语句:

1SELECT

2avg(number) median

3FROM

4( SELECT

5number, frequency, @beg1 := @end1+1 AS beg, @end1 := @beg1+frequency-1 AS end

6FROM

7numbers, (SELECT @beg1 := 0, @end1 :=0) init

8ORDER BY

9number

10) t1

11JOIN

12( SELECT

13number, frequency, @beg2 := @end2+1 AS beg, @end2 := @beg2+frequency-1 AS end

14FROM

15numbers, (SELECT @beg2 := 0, @end2 :=0) init

16ORDER BY

17number desc

18) t2

19using(number)

20WHERE

21t1.beg BETWEEN t2.beg-1 and t2.end+1 or t1.end BETWEEN t2.beg-1 and t2.end+1

查询效率:

我们发现,虽然解法3写起来相对复杂,但效率居然是最高的。不过个人还是比较喜欢解法2,即简单的根据中位数排序编号来判断,简单高效易懂。

以上就是LeetCode中两道关于中位数题目的几种解法,当然,肯定还有更多更好的解法,这里也只是简单探讨以作抛砖引玉。返回搜狐,查看更多

责任编辑:

Logo

魔乐社区(Modelers.cn) 是一个中立、公益的人工智能社区,提供人工智能工具、模型、数据的托管、展示与应用协同服务,为人工智能开发及爱好者搭建开放的学习交流平台。社区通过理事会方式运作,由全产业链共同建设、共同运营、共同享有,推动国产AI生态繁荣发展。

更多推荐