MySQL窗口函数OVER使用示例详细讲解

 更新时间:2023年01月05日 09:50:52   作者:开发老张  
这篇文章主要介绍了MySQL窗口函数OVER()用法及说明,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

窗口函数

OVER (PARTITION BY xxx ORDER BY xxx ASC/DESC)

测试数据表及数据

测试表 employee

CREATE TABLE employee (
    `id` int unsigned not null auto_increment primary key,
    `name` varchar(80),
    `age` int(11),
    `salary` DECIMAL(18,1),
    `dept_id` int(11)
) ENGINE=InnoDB default charset=utf8mb4;

插入测试数据

INSERT into employee values(3, '小肖', 29, 30000.0, 1);
INSERT into employee values(4, '小东', 30, 40000.0, 2);
INSERT into employee values(6, '小非', 24, 23456.0, 3);
INSERT into employee values(7, '晓飞', 30, 15000.0, 4);
INSERT into employee values(8, '小林', 23, 24000.0, null);
INSERT into employee values(10, '小五', 20, 4500.0, null);
INSERT into employee values(11, '张山', 24, 40000.0, 1);
INSERT into employee values(12, '小肖', 28, 35000.0, 2);
INSERT into employee values(13, '李四', 23, 50000.0, 1);
INSERT into employee values(17, '王武', 24, 56000.0, 2);
INSERT into employee values(18, '猪小屁', 2, 56000.0, 2);
INSERT into employee values(19, '小玉', 25, 58000.0, 1);
INSERT into employee values(21, '小张', 23, 50000.0, 1);
INSERT into employee values(22, '小胡', 25, 25000.0, 2);
INSERT into employee values(96, '小肖', 19, 35000.0, 1);
INSERT into employee values(97, '小林', 20, 20000.0, 2);

窗口函数

partition by 是分区,每个分区形成一个窗口,聚合等计算都在这个分区内完成;

order by 是排序,排完序的数据组成不同的窗口,不同值的数据组成不同的窗口;

空窗口

当窗口中为空时,就是对表中所有数据进行计算

mysql> select name,salary,SUM(salary) over() AS already_paid_salary FROM employee e ;

name|salary |already_paid_salary|
----+-------+-------------------+
小肖  |30000.0|           561956.0|
小东  |40000.0|           561956.0|
小非  |23456.0|           561956.0|
晓飞  |15000.0|           561956.0|
小林  |24000.0|           561956.0|
小五  | 4500.0|           561956.0|
张山  |40000.0|           561956.0|
小肖  |35000.0|           561956.0|
李四  |50000.0|           561956.0|
王武  |56000.0|           561956.0|
猪小屁 |56000.0|           561956.0|
小玉  |58000.0|           561956.0|
小张  |50000.0|           561956.0|
小胡  |25000.0|           561956.0|
小肖  |35000.0|           561956.0|
小林  |20000.0|           561956.0|

窗口中只有 ORDER BY

当窗口中只有 order by 时候,对全表数据进行排序,其作用和 FROM 后面的 ORDER BY 一样,

1)当与 FROM 后面的 ORDER BY 字段相同时,相当于只有 OVER(ORDER BY xxx)

mysql> select name,salary,SUM(salary) over(ORDER BY salary) AS already_paid_salary FROM employee e ;

name|salary |already_paid_salary|
----+-------+-------------------+
小五  | 4500.0|             4500.0|
晓飞  |15000.0|            19500.0|
小林  |20000.0|            39500.0|
小非  |23456.0|            62956.0|
小林  |24000.0|            86956.0|
小胡  |25000.0|           111956.0|
小肖  |30000.0|           141956.0|
小肖  |35000.0|           211956.0|
小肖  |35000.0|           211956.0|
小东  |40000.0|           291956.0|
张山  |40000.0|           291956.0|
李四  |50000.0|           391956.0|
小张  |50000.0|           391956.0|
王武  |56000.0|           503956.0|
猪小屁 |56000.0|           503956.0|
小玉  |58000.0|           561956.0|

2)当与 FROM 后面的 ORDER BY 字段不同时,FROM 子句的 ORDER BY 会覆盖 OVER() 中的 ORDER BY,FROM 子句中 ORDER BY 后值相同的才会按照 OVER() 子句中的 ORDER BY 排序;

mysql> select id,name,salary,SUM(salary) over(ORDER BY salary) AS already_paid_salary FROM employee e ORDER BY name;

id|name|salary |already_paid_salary|
--+----+-------+-------------------+
 4|小东  |40000.0|           291956.0|
10|小五  | 4500.0|             4500.0|
21|小张  |50000.0|           391956.0|
97|小林  |20000.0|            39500.0|
 8|小林  |24000.0|            86956.0|
19|小玉  |58000.0|           561956.0|
 3|小肖  |30000.0|           141956.0|
12|小肖  |35000.0|           211956.0|
96|小肖  |35000.0|           211956.0|
22|小胡  |25000.0|           111956.0|
 6|小非  |23456.0|            62956.0|
11|张山  |40000.0|           291956.0|
 7|晓飞  |15000.0|            19500.0|
13|李四  |50000.0|           391956.0|
18|猪小屁 |56000.0|           503956.0|
17|王武  |56000.0|           503956.0|

窗口中只有 PARTITION BY 时

此时的聚合函数会按照分组进行计算,分组内的所有行的数据都是这个分组中所有数据计算后的值;

mysql> select id,name,salary,dept_id,SUM(salary) over(PARTITION BY dept_id) AS already_paid_salary FROM employee;

id|name|salary |dept_id|already_paid_salary|
--+----+-------+-------+-------------------+
 8|小林  |24000.0|       |            28500.0|
10|小五  | 4500.0|       |            28500.0|
 3|小肖  |30000.0|      1|           263000.0|
11|张山  |40000.0|      1|           263000.0|
13|李四  |50000.0|      1|           263000.0|
19|小玉  |58000.0|      1|           263000.0|
21|小张  |50000.0|      1|           263000.0|
96|小肖  |35000.0|      1|           263000.0|
 4|小东  |40000.0|      2|           232000.0|
12|小肖  |35000.0|      2|           232000.0|
17|王武  |56000.0|      2|           232000.0|
18|猪小屁 |56000.0|      2|           232000.0|
22|小胡  |25000.0|      2|           232000.0|
97|小林  |20000.0|      2|           232000.0|
 6|小非  |23456.0|      3|            23456.0|
 7|晓飞  |15000.0|      4|            15000.0|

同时有 PARTITION BY 与 ORDER BY

ORDER BY 对 PARTITION BY 窗口中的数据进行排序,当 PARTITION BY 与 ORDER BY 列名不同时,聚合函数是根据排序进行逐个聚合计算的,当碰到 ORDER BY 相同的两个值时,同时计算两个值,并两行数据一致;当 PARTITION BY 与 ORDER BY 的列一致时,相当于只有 PARTITION BY;FROM 后面的 ORDER BY 是对整个表的数据进行排序,与 OVER 子句中的不同;当两者的字段不同时,先按照 OVER() 子句进行聚合计算,然后按照 FROM 子句的进行排序输出;

mysql> select id,name,salary,dept_id,SUM(salary) over(PARTITION BY dept_id ORDER BY name) AS already_paid_salary FROM employee e ;

id|name|salary |dept_id|already_paid_salary|
--+----+-------+-------+-------------------+
10|小五  | 4500.0|       |             4500.0|
 8|小林  |24000.0|       |            28500.0|
21|小张  |50000.0|      1|            50000.0|
19|小玉  |58000.0|      1|           108000.0|
 3|小肖  |30000.0|      1|           173000.0|
96|小肖  |35000.0|      1|           173000.0|
11|张山  |40000.0|      1|           213000.0|
13|李四  |50000.0|      1|           263000.0|
 4|小东  |40000.0|      2|            40000.0|
97|小林  |20000.0|      2|            60000.0|
12|小肖  |35000.0|      2|            95000.0|
22|小胡  |25000.0|      2|           120000.0|
18|猪小屁 |56000.0|      2|           176000.0
17|王武  |56000.0|      2|           232000.0|
 6|小非  |23456.0|      3|            23456.0|
 7|晓飞  |15000.0|      4|            15000.0|

添加 FROM 子句的 ORDER BY

mysql> select id,name,salary,dept_id,SUM(salary) over(PARTITION BY dept_id ORDER BY name) AS already_paid_salary FROM employee e ORDER BY name;

id|name|salary |dept_id|already_paid_salary|
--+----+-------+-------+-------------------+
 4|小东  |40000.0|      2|            40000.0|
10|小五  | 4500.0|       |             4500.0|
21|小张  |50000.0|      1|            50000.0|
 8|小林  |24000.0|       |            28500.0|
97|小林  |20000.0|      2|            60000.0|
19|小玉  |58000.0|      1|           108000.0|
 3|小肖  |30000.0|      1|           173000.0|
96|小肖  |35000.0|      1|           173000.0|
12|小肖  |35000.0|      2|            95000.0|
22|小胡  |25000.0|      2|           120000.0|
 6|小非  |23456.0|      3|            23456.0|
11|张山  |40000.0|      1|           213000.0|
 7|晓飞  |15000.0|      4|            15000.0|
13|李四  |50000.0|      1|           263000.0|
18|猪小屁 |56000.0|      2|           176000.0|
17|王武  |56000.0|      2|           232000.0|

PARTITION BY 与 ORDER BY 字段一致时,相当于只有 PARTITION BY:

mysql> select id,name,salary,dept_id,SUM(salary) over(PARTITION BY dept_id ORDER BY dept_id) AS already_paid_salary FROM employee;

id|name|salary |dept_id|already_paid_salary|
--+----+-------+-------+-------------------+
 8|小林  |24000.0|       |            28500.0|
10|小五  | 4500.0|       |            28500.0|
 3|小肖  |30000.0|      1|           263000.0|
11|张山  |40000.0|      1|           263000.0|
13|李四  |50000.0|      1|           263000.0|
19|小玉  |58000.0|      1|           263000.0|
21|小张  |50000.0|      1|           263000.0|
96|小肖  |35000.0|      1|           263000.0|
 4|小东  |40000.0|      2|           232000.0|
12|小肖  |35000.0|      2|           232000.0|
17|王武  |56000.0|      2|           232000.0|
18|猪小屁 |56000.0|      2|           232000.0
22|小胡  |25000.0|      2|           232000.0|
97|小林  |20000.0|      2|           232000.0|
 6|小非  |23456.0|      3|            23456.0|
 7|晓飞  |15000.0|      4|            15000.0|

到此这篇关于MySQL窗口函数OVER使用示例详细讲解的文章就介绍到这了,更多相关MySQL窗口函数内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • mysql安装报错unknown variable mysqlx_port=0.0

    mysql安装报错unknown variable mysqlx_port=0.0

    本文主要介绍了mysql安装报错unknown variable mysqlx_port=0.0,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-06-06
  • MySQL 逻辑备份与恢复测试的相关总结

    MySQL 逻辑备份与恢复测试的相关总结

    数据库逻辑备份就是备份软件按照我们最初所设计的逻辑关系,以数据库的逻辑结构对象为单位,将数据库中的数据按照预定义的逻辑关联格式一条一条生成相关的文本文件,以达到备份的目的。本文将具体介绍MySQL 逻辑备份的相关概念及如何做恢复测试。
    2021-05-05
  • MySql使用skip-name-resolve解决外网链接客户端过慢问题

    MySql使用skip-name-resolve解决外网链接客户端过慢问题

    在腾讯云上面搭建的mysql使用开发的电脑上navicat进行访问时总是特别的慢,原来是Mysql会对请求的地址进行域名解析,开发的电脑并没有域名,所以会导致特别的慢,下面通过本文给大家分享MySql使用skip-name-resolve解决外网链接客户端过慢问题
    2017-07-07
  • 当面试官问mysql中char与varchar的区别

    当面试官问mysql中char与varchar的区别

    这篇文章主要以聊天形式图片的添加,将面试官面试真实场景体现出来,好奇的朋友不要错过奥
    2021-08-08
  • MySQL之JSON类型字段的使用技巧分享

    MySQL之JSON类型字段的使用技巧分享

    这篇文章主要介绍了MySQL之JSON类型字段的使用技巧,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • Mysql数据库性能优化之子查询

    Mysql数据库性能优化之子查询

    这篇文章主要介绍了Mysql数据库性能优化之子查询的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-01-01
  • mysql如何让自增id归0解决方案

    mysql如何让自增id归0解决方案

    数据库的Id自增越来越大,要让自增重新从1开始:那么就用下面的方法吧
    2012-11-11
  • MySQL删除表的外键约束图文教程(简单易懂)

    MySQL删除表的外键约束图文教程(简单易懂)

    删除表不是特别常用,特别是对于存在外键关联的表,删除更得小心,这篇文章主要给大家介绍了关于MySQL删除表的外键约束的相关资料,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2024-07-07
  • 关于MySQL的时间进位问题浅析

    关于MySQL的时间进位问题浅析

    这篇文章主要给大家介绍了关于MySQL的时间进位问题的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用MySQL具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-12-12
  • MySQL六种约束的示例详解(全网最全)

    MySQL六种约束的示例详解(全网最全)

    约束是作用于表中字段上的规则,用于限制存储在表中的数据。这篇文章主要为大家整理了一下MySQL中六种约束的用法,感兴趣的可以了解一下
    2022-07-07

最新评论